blob: 4c6c21567a63bcf1df5d1bf63344f3eecabce4dc [file] [log] [blame]
Colin Cross4d9c2d12016-07-29 12:48:20 -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 (
Inseob Kim69378442019-06-03 19:10:47 +090018 "fmt"
Logan Chien41eabe62019-04-10 13:33:58 +080019 "io"
Jiyong Parkf1194352019-02-25 11:05:47 +090020 "path/filepath"
Jiyong Parkda732bd2018-11-02 18:23:15 +090021 "regexp"
Colin Crossadd04a82024-05-22 09:57:59 -070022 "slices"
Jooyung Han11b0fbd2021-02-05 02:28:22 +090023 "strconv"
Colin Cross4d9c2d12016-07-29 12:48:20 -070024 "strings"
Jiyong Parkda732bd2018-11-02 18:23:15 +090025 "sync"
Colin Cross4d9c2d12016-07-29 12:48:20 -070026
Colin Cross4d9c2d12016-07-29 12:48:20 -070027 "android/soong/android"
Kiyoung Kimaa394802024-01-08 12:55:45 +090028
Chris Parsons51f8c392021-08-03 21:01:05 -040029 "github.com/google/blueprint"
Colin Crossa14fb6a2024-10-23 16:57:06 -070030 "github.com/google/blueprint/depset"
Chris Parsons51f8c392021-08-03 21:01:05 -040031 "github.com/google/blueprint/pathtools"
Spandan Das4238c652022-09-09 01:38:47 +000032 "github.com/google/blueprint/proptools"
Colin Cross4d9c2d12016-07-29 12:48:20 -070033)
34
Liz Kammer2b376bc2022-01-12 12:00:49 -050035// LibraryProperties is a collection of properties shared by cc library rules/cc.
Colin Crossb916a382016-07-29 17:28:03 -070036type LibraryProperties struct {
Jerome Gaillard9b454732024-12-10 18:07:19 +000037 // local file name to pass to the linker as -exported_symbols_list
38 Exported_symbols_list *string `android:"path,arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070039 // local file name to pass to the linker as -unexported_symbols_list
Colin Cross27b922f2019-03-04 22:35:41 -080040 Unexported_symbols_list *string `android:"path,arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070041 // local file name to pass to the linker as -force_symbols_not_weak_list
Colin Cross27b922f2019-03-04 22:35:41 -080042 Force_symbols_not_weak_list *string `android:"path,arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070043 // local file name to pass to the linker as -force_symbols_weak_list
Colin Cross27b922f2019-03-04 22:35:41 -080044 Force_symbols_weak_list *string `android:"path,arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070045
46 // rename host libraries to prevent overlap with system installed libraries
47 Unique_host_soname *bool
48
Dan Willemsene1240db2016-11-03 14:28:51 -070049 Aidl struct {
50 // export headers generated from .aidl sources
Nan Zhang0007d812017-11-07 10:57:05 -080051 Export_aidl_headers *bool
Dan Willemsene1240db2016-11-03 14:28:51 -070052 }
53
Colin Cross0c461f12016-10-20 16:11:43 -070054 Proto struct {
55 // export headers generated from .proto sources
Nan Zhang0007d812017-11-07 10:57:05 -080056 Export_proto_headers *bool
Colin Cross0c461f12016-10-20 16:11:43 -070057 }
Dan Albertf563d252017-10-13 00:29:00 -070058
Inseob Kimc0907f12019-02-08 21:00:45 +090059 Sysprop struct {
60 // Whether platform owns this sysprop library.
61 Platform *bool
Inseob Kimb3f22ca2019-03-05 12:40:24 +090062 } `blueprint:"mutated"`
Inseob Kimc0907f12019-02-08 21:00:45 +090063
Nan Zhang0007d812017-11-07 10:57:05 -080064 Static_ndk_lib *bool
Jiyong Park7ed9de32018-10-15 22:25:07 +090065
Florian Mayer6dab96c2020-12-31 11:39:48 +000066 // Generate stubs to make this library accessible to APEXes.
Jiyong Park7ed9de32018-10-15 22:25:07 +090067 Stubs struct {
68 // Relative path to the symbol map. The symbol map provides the list of
69 // symbols that are exported for stubs variant of this library.
Spandan Das02be1012024-07-24 01:21:53 +000070 Symbol_file *string `android:"path,arch_variant"`
Jiyong Park7ed9de32018-10-15 22:25:07 +090071
Jiyong Parkd4a3a132021-03-17 20:21:35 +090072 // List versions to generate stubs libs for. The version name "current" is always
73 // implicitly added.
Jiyong Park7ed9de32018-10-15 22:25:07 +090074 Versions []string
Alan Stokes73feba32022-11-14 12:21:24 +000075
76 // Whether to not require the implementation of the library to be installed if a
77 // client of the stubs is installed. Defaults to true; set to false if the
78 // implementation is made available by some other means, e.g. in a Microdroid
79 // virtual machine.
80 Implementation_installable *bool
Spandan Das02be1012024-07-24 01:21:53 +000081 } `android:"arch_variant"`
dimitryd95964a2018-11-07 13:43:34 +010082
83 // set the name of the output
84 Stem *string `android:"arch_variant"`
85
Colin Cross0fd6a412019-08-16 14:22:10 -070086 // set suffix of the name of the output
87 Suffix *string `android:"arch_variant"`
88
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +080089 // Properties for ABI compatibility checker.
90 Header_abi_checker headerAbiCheckerProperties
91
Colin Cross0fd6a412019-08-16 14:22:10 -070092 Target struct {
Justin Yun63e9ec72020-10-29 16:49:43 +090093 Vendor, Product struct {
Colin Cross0fd6a412019-08-16 14:22:10 -070094 // set suffix of the name of the output
95 Suffix *string `android:"arch_variant"`
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +080096
97 Header_abi_checker headerAbiCheckerProperties
Jooyung Han85707de2023-12-01 14:21:13 +090098
99 // Disable stubs for vendor/product variants
100 // This is a workaround to keep `stubs` only for "core" variant (not product/vendor).
101 // It would be nice if we could put `stubs` into a `target: { core: {} }`
102 // block but it's not supported in soong yet. This could be removed/simplified once we have
103 // a better syntax.
104 No_stubs bool
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +0800105 }
106
107 Platform struct {
108 Header_abi_checker headerAbiCheckerProperties
Colin Cross0fd6a412019-08-16 14:22:10 -0700109 }
110 }
111
dimitryd95964a2018-11-07 13:43:34 +0100112 // Names of modules to be overridden. Listed modules can only be other shared libraries
113 // (in Make or Soong).
114 // This does not completely prevent installation of the overridden libraries, but if both
115 // binaries would be installed by default (in PRODUCT_PACKAGES) the other library will be removed
116 // from PRODUCT_PACKAGES.
117 Overrides []string
Logan Chiene3d7a0d2019-01-17 00:18:02 +0800118
Pete Bentley803e1612019-08-06 22:19:59 +0100119 // Inject boringssl hash into the shared library. This is only intended for use by external/boringssl.
120 Inject_bssl_hash *bool `android:"arch_variant"`
Colin Cross0477b422020-10-13 18:43:54 -0700121
Colin Cross127bb8b2020-12-16 16:46:01 -0800122 // If this is an LLNDK library, properties to describe the LLNDK stubs. Will be copied from
123 // the module pointed to by llndk_stubs if it is set.
Spandan Das02be1012024-07-24 01:21:53 +0000124 Llndk llndkLibraryProperties `android:"arch_variant"`
Colin Cross5271fea2021-04-27 13:06:04 -0700125
126 // If this is a vendor public library, properties to describe the vendor public library stubs.
127 Vendor_public_library vendorPublicLibraryProperties
Colin Crossa48ab5b2017-02-14 15:28:44 -0800128}
Colin Cross0c461f12016-10-20 16:11:43 -0700129
Chris Parsons3c27ca32020-11-20 12:42:07 -0500130// StaticProperties is a properties stanza to affect only attributes of the "static" variants of a
131// library module.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700132type StaticProperties struct {
133 Static StaticOrSharedProperties `android:"arch_variant"`
134}
135
Chris Parsons3c27ca32020-11-20 12:42:07 -0500136// SharedProperties is a properties stanza to affect only attributes of the "shared" variants of a
137// library module.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700138type SharedProperties struct {
139 Shared StaticOrSharedProperties `android:"arch_variant"`
140}
141
Chris Parsons3c27ca32020-11-20 12:42:07 -0500142// StaticOrSharedProperties is an embedded struct representing properties to affect attributes of
143// either only the "static" variants or only the "shared" variants of a library module. These override
144// the base properties of the same name.
145// Use `StaticProperties` or `SharedProperties`, depending on which variant is needed.
146// `StaticOrSharedProperties` exists only to avoid duplication.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700147type StaticOrSharedProperties struct {
Cole Faust96a692b2024-08-08 14:47:51 -0700148 Srcs proptools.Configurable[[]string] `android:"path,arch_variant"`
Evgenii Stepanov2080bfe2020-07-24 15:35:40 -0700149
Chih-Hung Hsieh769a51c2021-09-17 17:18:39 -0700150 Tidy_disabled_srcs []string `android:"path,arch_variant"`
151
Chih-Hung Hsieh9db8a0c2022-02-17 12:54:45 -0800152 Tidy_timeout_srcs []string `android:"path,arch_variant"`
153
Evgenii Stepanov2080bfe2020-07-24 15:35:40 -0700154 Sanitized Sanitized `android:"arch_variant"`
155
Cole Fauste96c16a2024-06-13 14:51:14 -0700156 Cflags proptools.Configurable[[]string] `android:"arch_variant"`
Colin Crosse1bb5d02019-09-24 14:55:04 -0700157
Cole Faustf0006e72024-08-19 14:39:19 -0700158 Enabled *bool `android:"arch_variant"`
159 Whole_static_libs proptools.Configurable[[]string] `android:"arch_variant"`
160 Static_libs proptools.Configurable[[]string] `android:"arch_variant"`
161 Shared_libs proptools.Configurable[[]string] `android:"arch_variant"`
162 System_shared_libs []string `android:"arch_variant"`
Colin Crosse1bb5d02019-09-24 14:55:04 -0700163
164 Export_shared_lib_headers []string `android:"arch_variant"`
165 Export_static_lib_headers []string `android:"arch_variant"`
Jiyong Parka90ca002019-10-07 15:47:24 +0900166
167 Apex_available []string `android:"arch_variant"`
Colin Cross1bc94122021-10-28 13:25:54 -0700168
169 Installable *bool `android:"arch_variant"`
Colin Crosse1bb5d02019-09-24 14:55:04 -0700170}
171
Colin Crossa48ab5b2017-02-14 15:28:44 -0800172type LibraryMutatedProperties struct {
Colin Crossb916a382016-07-29 17:28:03 -0700173 // Build a static variant
174 BuildStatic bool `blueprint:"mutated"`
175 // Build a shared variant
176 BuildShared bool `blueprint:"mutated"`
177 // This variant is shared
178 VariantIsShared bool `blueprint:"mutated"`
179 // This variant is static
180 VariantIsStatic bool `blueprint:"mutated"`
Jiyong Park7ed9de32018-10-15 22:25:07 +0900181
182 // This variant is a stubs lib
183 BuildStubs bool `blueprint:"mutated"`
Jiyong Parkd4a3a132021-03-17 20:21:35 +0900184 // This variant is the latest version
185 IsLatestVersion bool `blueprint:"mutated"`
Jiyong Park7ed9de32018-10-15 22:25:07 +0900186 // Version of the stubs lib
187 StubsVersion string `blueprint:"mutated"`
Colin Crossd1f898e2020-08-18 18:35:15 -0700188 // List of all stubs versions associated with an implementation lib
189 AllStubsVersions []string `blueprint:"mutated"`
Colin Crossb916a382016-07-29 17:28:03 -0700190}
191
192type FlagExporterProperties struct {
193 // list of directories relative to the Blueprints file that will
Dan Willemsen273af7f2016-11-03 15:53:42 -0700194 // be added to the include path (using -I) for this module and any module that links
Colin Cross5d195602017-10-17 16:15:50 -0700195 // against this module. Directories listed in export_include_dirs do not need to be
196 // listed in local_include_dirs.
Aleks Todorovc9becde2024-06-10 12:51:53 +0100197 Export_include_dirs proptools.Configurable[[]string] `android:"arch_variant,variant_prepend"`
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700198
Jiyong Park73c54ee2019-10-22 20:31:18 +0900199 // list of directories that will be added to the system include path
200 // using -isystem for this module and any module that links against this module.
Colin Cross0ed579e2021-06-29 00:51:12 +0000201 Export_system_include_dirs []string `android:"arch_variant,variant_prepend"`
Jiyong Park73c54ee2019-10-22 20:31:18 +0900202
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700203 Target struct {
Justin Yun63e9ec72020-10-29 16:49:43 +0900204 Vendor, Product struct {
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700205 // list of exported include directories, like
Justin Yun63e9ec72020-10-29 16:49:43 +0900206 // export_include_dirs, that will be applied to
207 // vendor or product variant of this library.
208 // This will overwrite any other declarations.
Steven Morelandb21df8f2018-01-05 14:42:54 -0800209 Override_export_include_dirs []string
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700210 }
211 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700212}
213
214func init() {
Paul Duffin6c26dc72019-12-19 15:02:40 +0000215 RegisterLibraryBuildComponents(android.InitRegistrationContext)
216}
217
218func RegisterLibraryBuildComponents(ctx android.RegistrationContext) {
219 ctx.RegisterModuleType("cc_library_static", LibraryStaticFactory)
220 ctx.RegisterModuleType("cc_library_shared", LibrarySharedFactory)
221 ctx.RegisterModuleType("cc_library", LibraryFactory)
222 ctx.RegisterModuleType("cc_library_host_static", LibraryHostStaticFactory)
223 ctx.RegisterModuleType("cc_library_host_shared", LibraryHostSharedFactory)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700224}
225
Patrice Arruda83c89e02019-03-25 15:32:39 -0700226// cc_library creates both static and/or shared libraries for a device and/or
227// host. By default, a cc_library has a single variant that targets the device.
228// Specifying `host_supported: true` also creates a library that targets the
229// host.
Steven Morelandf9e62162017-11-02 17:00:50 -0700230func LibraryFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800231 module, _ := NewLibrary(android.HostAndDeviceSupported)
Paul Duffina0843f62019-12-13 19:50:38 +0000232 // Can be used as both a static and a shared library.
233 module.sdkMemberTypes = []android.SdkMemberType{
234 sharedLibrarySdkMemberType,
235 staticLibrarySdkMemberType,
Paul Duffin9b76c0b2020-03-12 10:24:35 +0000236 staticAndSharedLibrarySdkMemberType,
Paul Duffina0843f62019-12-13 19:50:38 +0000237 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700238 return module.Init()
239}
240
Patrice Arruda83c89e02019-03-25 15:32:39 -0700241// cc_library_static creates a static library for a device and/or host binary.
Steven Morelandf9e62162017-11-02 17:00:50 -0700242func LibraryStaticFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800243 module, library := NewLibrary(android.HostAndDeviceSupported)
244 library.BuildOnlyStatic()
Paul Duffina0843f62019-12-13 19:50:38 +0000245 module.sdkMemberTypes = []android.SdkMemberType{staticLibrarySdkMemberType}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700246 return module.Init()
247}
248
Patrice Arruda83c89e02019-03-25 15:32:39 -0700249// cc_library_shared creates a shared library for a device and/or host.
Steven Morelandf9e62162017-11-02 17:00:50 -0700250func LibrarySharedFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800251 module, library := NewLibrary(android.HostAndDeviceSupported)
252 library.BuildOnlyShared()
Paul Duffina0843f62019-12-13 19:50:38 +0000253 module.sdkMemberTypes = []android.SdkMemberType{sharedLibrarySdkMemberType}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700254 return module.Init()
255}
256
Patrice Arruda83c89e02019-03-25 15:32:39 -0700257// cc_library_host_static creates a static library that is linkable to a host
258// binary.
Steven Morelandf9e62162017-11-02 17:00:50 -0700259func LibraryHostStaticFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800260 module, library := NewLibrary(android.HostSupported)
261 library.BuildOnlyStatic()
Paul Duffina0843f62019-12-13 19:50:38 +0000262 module.sdkMemberTypes = []android.SdkMemberType{staticLibrarySdkMemberType}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700263 return module.Init()
264}
265
Patrice Arruda83c89e02019-03-25 15:32:39 -0700266// cc_library_host_shared creates a shared library that is usable on a host.
Steven Morelandf9e62162017-11-02 17:00:50 -0700267func LibraryHostSharedFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800268 module, library := NewLibrary(android.HostSupported)
269 library.BuildOnlyShared()
Paul Duffina0843f62019-12-13 19:50:38 +0000270 module.sdkMemberTypes = []android.SdkMemberType{sharedLibrarySdkMemberType}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700271 return module.Init()
272}
273
Chris Parsons3c27ca32020-11-20 12:42:07 -0500274// flagExporter is a separated portion of libraryDecorator pertaining to exported
275// include paths and flags. Keeping this dependency-related information separate
276// from the rest of library information is helpful in keeping data more structured
277// and explicit.
Colin Cross4d9c2d12016-07-29 12:48:20 -0700278type flagExporter struct {
279 Properties FlagExporterProperties
280
Ivan Lozano0a468a42024-05-13 21:03:34 -0400281 dirs android.Paths // Include directories to be included with -I
282 systemDirs android.Paths // System include directories to be included with -isystem
283 flags []string // Exported raw flags.
284 deps android.Paths
285 headers android.Paths
286 rustRlibDeps []RustRlibDep
Colin Cross4d9c2d12016-07-29 12:48:20 -0700287}
288
Chris Parsons3c27ca32020-11-20 12:42:07 -0500289// exportedIncludes returns the effective include paths for this module and
290// any module that links against this module. This is obtained from
291// the export_include_dirs property in the appropriate target stanza.
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700292func (f *flagExporter) exportedIncludes(ctx ModuleContext) android.Paths {
Justin Yun6977e8a2020-10-29 18:24:11 +0900293 if ctx.inVendor() && f.Properties.Target.Vendor.Override_export_include_dirs != nil {
Steven Morelandb21df8f2018-01-05 14:42:54 -0800294 return android.PathsForModuleSrc(ctx, f.Properties.Target.Vendor.Override_export_include_dirs)
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700295 }
Justin Yun6977e8a2020-10-29 18:24:11 +0900296 if ctx.inProduct() && f.Properties.Target.Product.Override_export_include_dirs != nil {
297 return android.PathsForModuleSrc(ctx, f.Properties.Target.Product.Override_export_include_dirs)
298 }
Aleks Todorovc9becde2024-06-10 12:51:53 +0100299 return android.PathsForModuleSrc(ctx, f.Properties.Export_include_dirs.GetOrDefault(ctx, nil))
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700300}
301
Hsin-Yi Chen5f228b02024-04-02 12:38:47 +0800302func (f *flagExporter) exportedSystemIncludes(ctx ModuleContext) android.Paths {
303 return android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)
304}
305
Chris Parsons3c27ca32020-11-20 12:42:07 -0500306// exportIncludes registers the include directories and system include directories to be exported
307// transitively to modules depending on this module.
Inseob Kim69378442019-06-03 19:10:47 +0900308func (f *flagExporter) exportIncludes(ctx ModuleContext) {
Jiyong Park74955042019-10-22 20:19:51 +0900309 f.dirs = append(f.dirs, f.exportedIncludes(ctx)...)
Jiyong Park73c54ee2019-10-22 20:31:18 +0900310 f.systemDirs = append(f.systemDirs, android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700311}
312
Chris Parsons3c27ca32020-11-20 12:42:07 -0500313// exportIncludesAsSystem registers the include directories and system include directories to be
314// exported transitively both as system include directories to modules depending on this module.
Inseob Kim69378442019-06-03 19:10:47 +0900315func (f *flagExporter) exportIncludesAsSystem(ctx ModuleContext) {
Jiyong Park73c54ee2019-10-22 20:31:18 +0900316 // all dirs are force exported as system
Jiyong Park74955042019-10-22 20:19:51 +0900317 f.systemDirs = append(f.systemDirs, f.exportedIncludes(ctx)...)
Jiyong Park73c54ee2019-10-22 20:31:18 +0900318 f.systemDirs = append(f.systemDirs, android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)...)
Inseob Kim69378442019-06-03 19:10:47 +0900319}
320
Chris Parsons3c27ca32020-11-20 12:42:07 -0500321// reexportDirs registers the given directories as include directories to be exported transitively
322// to modules depending on this module.
Jiyong Park74955042019-10-22 20:19:51 +0900323func (f *flagExporter) reexportDirs(dirs ...android.Path) {
Inseob Kim69378442019-06-03 19:10:47 +0900324 f.dirs = append(f.dirs, dirs...)
325}
326
Chris Parsons3c27ca32020-11-20 12:42:07 -0500327// reexportSystemDirs registers the given directories as system include directories
328// to be exported transitively to modules depending on this module.
Jiyong Park74955042019-10-22 20:19:51 +0900329func (f *flagExporter) reexportSystemDirs(dirs ...android.Path) {
Inseob Kim69378442019-06-03 19:10:47 +0900330 f.systemDirs = append(f.systemDirs, dirs...)
331}
332
Chris Parsons3c27ca32020-11-20 12:42:07 -0500333// reexportFlags registers the flags to be exported transitively to modules depending on this
334// module.
Inseob Kim69378442019-06-03 19:10:47 +0900335func (f *flagExporter) reexportFlags(flags ...string) {
Jaewoong Jung3aff5782020-02-11 07:54:35 -0800336 if android.PrefixInList(flags, "-I") || android.PrefixInList(flags, "-isystem") {
337 panic(fmt.Errorf("Exporting invalid flag %q: "+
338 "use reexportDirs or reexportSystemDirs to export directories", flag))
Inseob Kim69378442019-06-03 19:10:47 +0900339 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700340 f.flags = append(f.flags, flags...)
341}
342
Inseob Kim69378442019-06-03 19:10:47 +0900343func (f *flagExporter) reexportDeps(deps ...android.Path) {
344 f.deps = append(f.deps, deps...)
345}
346
Ivan Lozano0a468a42024-05-13 21:03:34 -0400347func (f *flagExporter) reexportRustStaticDeps(deps ...RustRlibDep) {
348 f.rustRlibDeps = append(f.rustRlibDeps, deps...)
349}
350
Inseob Kimd110f872019-12-06 13:15:38 +0900351// addExportedGeneratedHeaders does nothing but collects generated header files.
352// This can be differ to exportedDeps which may contain phony files to minimize ninja.
353func (f *flagExporter) addExportedGeneratedHeaders(headers ...android.Path) {
354 f.headers = append(f.headers, headers...)
355}
356
Colin Cross0de8a1e2020-09-18 14:15:30 -0700357func (f *flagExporter) setProvider(ctx android.ModuleContext) {
Colin Cross40213022023-12-13 15:19:49 -0800358 android.SetProvider(ctx, FlagExporterInfoProvider, FlagExporterInfo{
Chris Parsonsf60ecf02021-04-27 14:48:30 -0400359 // Comes from Export_include_dirs property, and those of exported transitive deps
360 IncludeDirs: android.FirstUniquePaths(f.dirs),
361 // Comes from Export_system_include_dirs property, and those of exported transitive deps
Colin Crossc82e6e22021-04-20 18:21:50 -0700362 SystemIncludeDirs: android.FirstUniquePaths(f.systemDirs),
Chris Parsonsf60ecf02021-04-27 14:48:30 -0400363 // Used in very few places as a one-off way of adding extra defines.
364 Flags: f.flags,
365 // Used sparingly, for extra files that need to be explicitly exported to dependers,
366 // or for phony files to minimize ninja.
367 Deps: f.deps,
Ivan Lozano0a468a42024-05-13 21:03:34 -0400368 // Used for exporting rlib deps of static libraries to dependents.
369 RustRlibDeps: f.rustRlibDeps,
Chris Parsonsf60ecf02021-04-27 14:48:30 -0400370 // For exported generated headers, such as exported aidl headers, proto headers, or
371 // sysprop headers.
372 GeneratedHeaders: f.headers,
Colin Cross0de8a1e2020-09-18 14:15:30 -0700373 })
Inseob Kim69378442019-06-03 19:10:47 +0900374}
375
Colin Crossb916a382016-07-29 17:28:03 -0700376// libraryDecorator wraps baseCompiler, baseLinker and baseInstaller to provide library-specific
377// functionality: static vs. shared linkage, reusing object files for shared libraries
378type libraryDecorator struct {
Colin Crossa48ab5b2017-02-14 15:28:44 -0800379 Properties LibraryProperties
Colin Crosse1bb5d02019-09-24 14:55:04 -0700380 StaticProperties StaticProperties
381 SharedProperties SharedProperties
Colin Crossa48ab5b2017-02-14 15:28:44 -0800382 MutatedProperties LibraryMutatedProperties
Colin Cross4d9c2d12016-07-29 12:48:20 -0700383
384 // For reusing static library objects for shared library
Inseob Kim69378442019-06-03 19:10:47 +0900385 reuseObjects Objects
Colin Cross10d22312017-05-03 11:01:58 -0700386
Colin Cross26c34ed2016-09-30 17:10:16 -0700387 // table-of-contents file to optimize out relinking when possible
388 tocFile android.OptionalPath
Colin Cross4d9c2d12016-07-29 12:48:20 -0700389
Colin Cross4d9c2d12016-07-29 12:48:20 -0700390 flagExporter
Liz Kammerb6a55bf2021-04-12 15:42:51 -0400391 flagExporterInfo *FlagExporterInfo
392 stripper Stripper
Colin Cross4d9c2d12016-07-29 12:48:20 -0700393
Colin Cross4d9c2d12016-07-29 12:48:20 -0700394 // For whole_static_libs
Colin Crossa2bcf2c2022-02-11 13:11:55 -0800395 objects Objects
396 wholeStaticLibsFromPrebuilts android.Paths
Colin Cross4d9c2d12016-07-29 12:48:20 -0700397
398 // Uses the module's name if empty, but can be overridden. Does not include
399 // shlib suffix.
400 libName string
Colin Crossb916a382016-07-29 17:28:03 -0700401
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -0800402 sabi *sabi
403
Dan Willemsen581341d2017-02-09 16:16:31 -0800404 // Output archive of gcno coverage information files
405 coverageOutputFile android.OptionalPath
406
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -0800407 // Source Abi Diff
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +0800408 sAbiDiff android.Paths
Mu-Le Lee0a9005e2022-07-05 09:49:50 +0000409
Colin Cross0875c522017-11-28 17:34:01 -0800410 // Location of the static library in the sysroot. Empty if the library is
411 // not included in the NDK.
412 ndkSysrootPath android.Path
413
Colin Crossb60190a2018-09-04 16:28:17 -0700414 // Location of the linked, unstripped library for shared libraries
415 unstrippedOutputFile android.Path
Wei Li5f5d2712023-12-11 15:40:29 -0800416 // Location of the linked, stripped library for shared libraries, strip: "all"
417 strippedAllOutputFile android.Path
Colin Crossb60190a2018-09-04 16:28:17 -0700418
Dan Willemsen569edc52018-11-19 09:33:29 -0800419 // Location of the file that should be copied to dist dir when requested
Jingwen Chen40fd90a2020-06-15 05:24:19 +0000420 distFile android.Path
Dan Willemsen569edc52018-11-19 09:33:29 -0800421
Colin Cross8e21aa52020-09-28 18:28:02 -0700422 versionScriptPath android.OptionalPath
Jiyong Park7ed9de32018-10-15 22:25:07 +0900423
Jaewoong Jung18aefc12020-12-21 09:11:10 -0800424 postInstallCmds []string
Jiyong Parkf1194352019-02-25 11:05:47 +0900425
Colin Cross5ec407b2020-09-30 11:41:33 -0700426 skipAPIDefine bool
427
Martin Stjernholmc5dd4f72020-04-01 20:38:01 +0100428 // Decorated interfaces
Colin Crossb916a382016-07-29 17:28:03 -0700429 *baseCompiler
430 *baseLinker
431 *baseInstaller
Inseob Kimeda2e9c2020-03-03 22:06:32 +0900432
sophiez4c4f8032021-08-16 22:54:00 -0700433 apiListCoverageXmlPath android.ModuleOutPath
Spandan Dase20c56c2024-07-23 21:34:24 +0000434
435 // Path to the file containing the APIs exported by this library
436 stubsSymbolFilePath android.Path
Inseob Kimeda2e9c2020-03-03 22:06:32 +0900437}
438
Chris Parsons3c27ca32020-11-20 12:42:07 -0500439// linkerProps returns the list of properties structs relevant for this library. (For example, if
440// the library is cc_shared_library, then static-library properties are omitted.)
Colin Crossb916a382016-07-29 17:28:03 -0700441func (library *libraryDecorator) linkerProps() []interface{} {
442 var props []interface{}
443 props = append(props, library.baseLinker.linkerProps()...)
Colin Crosse1bb5d02019-09-24 14:55:04 -0700444 props = append(props,
Colin Cross4d9c2d12016-07-29 12:48:20 -0700445 &library.Properties,
Colin Crossa48ab5b2017-02-14 15:28:44 -0800446 &library.MutatedProperties,
Colin Cross4d9c2d12016-07-29 12:48:20 -0700447 &library.flagExporter.Properties,
Colin Cross22f37952018-09-05 10:43:13 -0700448 &library.stripper.StripProperties)
Colin Crosse1bb5d02019-09-24 14:55:04 -0700449
450 if library.MutatedProperties.BuildShared {
451 props = append(props, &library.SharedProperties)
452 }
453 if library.MutatedProperties.BuildStatic {
454 props = append(props, &library.StaticProperties)
455 }
456
457 return props
Colin Cross4d9c2d12016-07-29 12:48:20 -0700458}
459
Chris Parsons3c27ca32020-11-20 12:42:07 -0500460// linkerFlags takes a Flags struct and augments it to contain linker flags that are defined by this
461// library, or that are implied by attributes of this library (such as whether this library is a
462// shared library).
Colin Crossb916a382016-07-29 17:28:03 -0700463func (library *libraryDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Colin Cross42742b82016-08-01 13:20:05 -0700464 flags = library.baseLinker.linkerFlags(ctx, flags)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700465
Colin Crossb916a382016-07-29 17:28:03 -0700466 // MinGW spits out warnings about -fPIC even for -fpie?!) being ignored because
467 // all code is position independent, and then those warnings get promoted to
468 // errors.
Colin Cross3edeee12017-04-04 12:59:48 -0700469 if !ctx.Windows() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800470 flags.Global.CFlags = append(flags.Global.CFlags, "-fPIC")
Colin Crossb916a382016-07-29 17:28:03 -0700471 }
472
473 if library.static() {
Cole Fauste96c16a2024-06-13 14:51:14 -0700474 flags.Local.CFlags = append(flags.Local.CFlags, library.StaticProperties.Static.Cflags.GetOrDefault(ctx, nil)...)
Colin Crossa48ab5b2017-02-14 15:28:44 -0800475 } else if library.shared() {
Cole Fauste96c16a2024-06-13 14:51:14 -0700476 flags.Local.CFlags = append(flags.Local.CFlags, library.SharedProperties.Shared.Cflags.GetOrDefault(ctx, nil)...)
Colin Crossb916a382016-07-29 17:28:03 -0700477 }
478
Colin Crossa48ab5b2017-02-14 15:28:44 -0800479 if library.shared() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700480 libName := library.getLibName(ctx)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700481 var f []string
Dan Willemsen01a405a2016-06-13 17:19:03 -0700482 if ctx.toolchain().Bionic() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700483 f = append(f,
484 "-nostdlib",
485 "-Wl,--gc-sections",
486 )
487 }
488
489 if ctx.Darwin() {
490 f = append(f,
491 "-dynamiclib",
Colin Cross4d9c2d12016-07-29 12:48:20 -0700492 "-install_name @rpath/"+libName+flags.Toolchain.ShlibSuffix(),
493 )
Colin Cross7863cf52016-10-20 10:47:21 -0700494 if ctx.Arch().ArchType == android.X86 {
495 f = append(f,
496 "-read_only_relocs suppress",
497 )
498 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700499 } else {
Josh Gao75a50a22019-06-07 17:58:59 -0700500 f = append(f, "-shared")
501 if !ctx.Windows() {
502 f = append(f, "-Wl,-soname,"+libName+flags.Toolchain.ShlibSuffix())
503 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700504 }
505
Colin Cross4af21ed2019-11-04 09:37:55 -0800506 flags.Global.LdFlags = append(flags.Global.LdFlags, f...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700507 }
508
509 return flags
510}
511
Chris Parsons3c27ca32020-11-20 12:42:07 -0500512// compilerFlags takes a Flags and augments it to contain compile flags from global values,
513// per-target values, module type values, per-module Blueprints properties, extra flags from
514// `flags`, and generated sources from `deps`.
Colin Crossf18e1102017-11-16 14:33:08 -0800515func (library *libraryDecorator) compilerFlags(ctx ModuleContext, flags Flags, deps PathDeps) Flags {
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700516 exportIncludeDirs := library.flagExporter.exportedIncludes(ctx)
Dan Willemsen273af7f2016-11-03 15:53:42 -0700517 if len(exportIncludeDirs) > 0 {
Colin Crossdad8c952017-04-26 14:55:27 -0700518 f := includeDirsToFlags(exportIncludeDirs)
Colin Cross4af21ed2019-11-04 09:37:55 -0800519 flags.Local.CommonFlags = append(flags.Local.CommonFlags, f)
520 flags.Local.YasmFlags = append(flags.Local.YasmFlags, f)
Dan Willemsen273af7f2016-11-03 15:53:42 -0700521 }
522
Jiyong Park7ed9de32018-10-15 22:25:07 +0900523 flags = library.baseCompiler.compilerFlags(ctx, flags, deps)
Colin Cross127bb8b2020-12-16 16:46:01 -0800524 if ctx.IsLlndk() {
525 // LLNDK libraries ignore most of the properties on the cc_library and use the
526 // LLNDK-specific properties instead.
527 // Wipe all the module-local properties, leaving only the global properties.
528 flags.Local = LocalOrGlobalFlags{}
529 }
Jiyong Park7ed9de32018-10-15 22:25:07 +0900530 if library.buildStubs() {
Jiyong Park64379952018-12-13 18:37:29 +0900531 // Remove -include <file> when compiling stubs. Otherwise, the force included
532 // headers might cause conflicting types error with the symbols in the
533 // generated stubs source code. e.g.
534 // double acos(double); // in header
535 // void acos() {} // in the generated source code
536 removeInclude := func(flags []string) []string {
537 ret := flags[:0]
538 for _, f := range flags {
539 if strings.HasPrefix(f, "-include ") {
540 continue
541 }
542 ret = append(ret, f)
543 }
544 return ret
545 }
Colin Cross4af21ed2019-11-04 09:37:55 -0800546 flags.Local.CommonFlags = removeInclude(flags.Local.CommonFlags)
547 flags.Local.CFlags = removeInclude(flags.Local.CFlags)
Jiyong Park64379952018-12-13 18:37:29 +0900548
Jiyong Park7ed9de32018-10-15 22:25:07 +0900549 flags = addStubLibraryCompilerFlags(flags)
550 }
551 return flags
Dan Willemsen273af7f2016-11-03 15:53:42 -0700552}
553
Colin Cross91ae5ec2024-10-01 14:03:40 -0700554func (library *libraryDecorator) getHeaderAbiCheckerProperties(m *Module) headerAbiCheckerProperties {
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +0800555 variantProps := &library.Properties.Target.Platform.Header_abi_checker
556 if m.InVendor() {
557 variantProps = &library.Properties.Target.Vendor.Header_abi_checker
558 } else if m.InProduct() {
559 variantProps = &library.Properties.Target.Product.Header_abi_checker
560 }
561 props := library.Properties.Header_abi_checker
562 err := proptools.AppendProperties(&props, variantProps, nil)
563 if err != nil {
Colin Cross91ae5ec2024-10-01 14:03:40 -0700564 panic(fmt.Errorf("Cannot merge headerAbiCheckerProperties: %s", err.Error()))
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +0800565 }
566 return props
Logan Chien41eabe62019-04-10 13:33:58 +0800567}
568
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700569func (library *libraryDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
Colin Cross127bb8b2020-12-16 16:46:01 -0800570 if ctx.IsLlndk() {
Justin Yun22abf212024-10-10 16:22:09 +0900571 // Get the matching SDK version for the vendor API level.
572 version, err := android.GetSdkVersionForVendorApiLevel(ctx.Config().VendorApiLevel())
573 if err != nil {
574 panic(err)
575 }
576
577 // This is the vendor variant of an LLNDK library, build the LLNDK stubs.
Dan Albertf1d14c72020-07-30 14:32:55 -0700578 nativeAbiResult := parseNativeAbiDefinition(ctx,
579 String(library.Properties.Llndk.Symbol_file),
Justin Yun22abf212024-10-10 16:22:09 +0900580 nativeClampedApiLevel(ctx, version), "--llndk")
Dan Albertf1d14c72020-07-30 14:32:55 -0700581 objs := compileStubLibrary(ctx, flags, nativeAbiResult.stubSrc)
Colin Cross127bb8b2020-12-16 16:46:01 -0800582 if !Bool(library.Properties.Llndk.Unversioned) {
Dan Albertf1d14c72020-07-30 14:32:55 -0700583 library.versionScriptPath = android.OptionalPathForPath(
584 nativeAbiResult.versionScript)
Colin Cross127bb8b2020-12-16 16:46:01 -0800585 }
586 return objs
587 }
Colin Cross5271fea2021-04-27 13:06:04 -0700588 if ctx.IsVendorPublicLibrary() {
Dan Albertf1d14c72020-07-30 14:32:55 -0700589 nativeAbiResult := parseNativeAbiDefinition(ctx,
590 String(library.Properties.Vendor_public_library.Symbol_file),
591 android.FutureApiLevel, "")
592 objs := compileStubLibrary(ctx, flags, nativeAbiResult.stubSrc)
Colin Cross5271fea2021-04-27 13:06:04 -0700593 if !Bool(library.Properties.Vendor_public_library.Unversioned) {
Dan Albertf1d14c72020-07-30 14:32:55 -0700594 library.versionScriptPath = android.OptionalPathForPath(nativeAbiResult.versionScript)
Colin Cross5271fea2021-04-27 13:06:04 -0700595 }
596 return objs
597 }
Jiyong Park7ed9de32018-10-15 22:25:07 +0900598 if library.buildStubs() {
Spandan Das357ffcc2024-07-24 18:07:48 +0000599 return library.compileModuleLibApiStubs(ctx, flags, deps)
Jiyong Park7ed9de32018-10-15 22:25:07 +0900600 }
601
Cole Faust96a692b2024-08-08 14:47:51 -0700602 srcs := library.baseCompiler.Properties.Srcs.GetOrDefault(ctx, nil)
603 staticSrcs := library.StaticProperties.Static.Srcs.GetOrDefault(ctx, nil)
604 sharedSrcs := library.SharedProperties.Shared.Srcs.GetOrDefault(ctx, nil)
Colin Cross5950f382016-12-13 12:50:57 -0800605 if !library.buildShared() && !library.buildStatic() {
Cole Faust96a692b2024-08-08 14:47:51 -0700606 if len(srcs) > 0 {
Colin Cross5950f382016-12-13 12:50:57 -0800607 ctx.PropertyErrorf("srcs", "cc_library_headers must not have any srcs")
608 }
Cole Faust96a692b2024-08-08 14:47:51 -0700609 if len(staticSrcs) > 0 {
Colin Cross5950f382016-12-13 12:50:57 -0800610 ctx.PropertyErrorf("static.srcs", "cc_library_headers must not have any srcs")
611 }
Cole Faust96a692b2024-08-08 14:47:51 -0700612 if len(sharedSrcs) > 0 {
Colin Cross5950f382016-12-13 12:50:57 -0800613 ctx.PropertyErrorf("shared.srcs", "cc_library_headers must not have any srcs")
614 }
615 return Objects{}
616 }
Yo Chiang2bbadfa2020-12-14 11:42:16 +0800617 if library.sabi.shouldCreateSourceAbiDump() {
Hsin-Yi Chenaf369882024-03-29 20:10:36 +0800618 dirs := library.exportedIncludeDirsForAbiCheck(ctx)
Hsin-Yi Chenea748242025-01-15 15:44:48 +0800619 flags.SAbiFlags = make([]string, 0, len(dirs)+1)
Hsin-Yi Chenaf369882024-03-29 20:10:36 +0800620 for _, dir := range dirs {
621 flags.SAbiFlags = append(flags.SAbiFlags, "-I"+dir)
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -0800622 }
Hsin-Yi Chenea748242025-01-15 15:44:48 +0800623 // If this library does not export any include directory, do not append the flags
624 // so that the ABI tool dumps everything without filtering by the include directories.
625 // requiresGlobalIncludes returns whether this library can include CommonGlobalIncludes.
626 // If the library cannot include them, it cannot export them.
627 if len(dirs) > 0 && requiresGlobalIncludes(ctx) {
628 flags.SAbiFlags = append(flags.SAbiFlags, "${config.CommonGlobalIncludes}")
629 }
Cole Faust96a692b2024-08-08 14:47:51 -0700630 totalLength := len(srcs) + len(deps.GeneratedSources) +
631 len(sharedSrcs) + len(staticSrcs)
Jaewoong Jung18aefc12020-12-21 09:11:10 -0800632 if totalLength > 0 {
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -0800633 flags.SAbiDump = true
634 }
635 }
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700636 objs := library.baseCompiler.compile(ctx, flags, deps)
637 library.reuseObjects = objs
Colin Cross2f336352016-10-26 10:03:47 -0700638 buildFlags := flagsToBuilderFlags(flags)
Colin Crossb916a382016-07-29 17:28:03 -0700639
Colin Cross4d9c2d12016-07-29 12:48:20 -0700640 if library.static() {
Cole Faust96a692b2024-08-08 14:47:51 -0700641 srcs := android.PathsForModuleSrc(ctx, staticSrcs)
Chih-Hung Hsieh769a51c2021-09-17 17:18:39 -0700642 objs = objs.Append(compileObjs(ctx, buildFlags, android.DeviceStaticLibrary, srcs,
643 android.PathsForModuleSrc(ctx, library.StaticProperties.Static.Tidy_disabled_srcs),
Chih-Hung Hsieh9db8a0c2022-02-17 12:54:45 -0800644 android.PathsForModuleSrc(ctx, library.StaticProperties.Static.Tidy_timeout_srcs),
Chih-Hung Hsieh769a51c2021-09-17 17:18:39 -0700645 library.baseCompiler.pathDeps, library.baseCompiler.cFlagsDeps))
Colin Crossa48ab5b2017-02-14 15:28:44 -0800646 } else if library.shared() {
Cole Faust96a692b2024-08-08 14:47:51 -0700647 srcs := android.PathsForModuleSrc(ctx, sharedSrcs)
Chih-Hung Hsieh769a51c2021-09-17 17:18:39 -0700648 objs = objs.Append(compileObjs(ctx, buildFlags, android.DeviceSharedLibrary, srcs,
649 android.PathsForModuleSrc(ctx, library.SharedProperties.Shared.Tidy_disabled_srcs),
Chih-Hung Hsieh9db8a0c2022-02-17 12:54:45 -0800650 android.PathsForModuleSrc(ctx, library.SharedProperties.Shared.Tidy_timeout_srcs),
Chih-Hung Hsieh769a51c2021-09-17 17:18:39 -0700651 library.baseCompiler.pathDeps, library.baseCompiler.cFlagsDeps))
Colin Crossb916a382016-07-29 17:28:03 -0700652 }
653
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700654 return objs
Colin Crossb916a382016-07-29 17:28:03 -0700655}
656
Spandan Das357ffcc2024-07-24 18:07:48 +0000657// Compile stubs for the API surface between platform and apex
658// This method will be used by source and prebuilt cc module types.
659func (library *libraryDecorator) compileModuleLibApiStubs(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
660 // TODO (b/275273834): Make this a hard error when the symbol files have been added to module sdk.
661 if library.Properties.Stubs.Symbol_file == nil {
662 return Objects{}
663 }
664 symbolFile := String(library.Properties.Stubs.Symbol_file)
665 library.stubsSymbolFilePath = android.PathForModuleSrc(ctx, symbolFile)
666 // b/239274367 --apex and --systemapi filters symbols tagged with # apex and #
667 // systemapi, respectively. The former is for symbols defined in platform libraries
668 // and the latter is for symbols defined in APEXes.
669 // A single library can contain either # apex or # systemapi, but not both.
670 // The stub generator (ndkstubgen) is additive, so passing _both_ of these to it should be a no-op.
671 // However, having this distinction helps guard accidental
672 // promotion or demotion of API and also helps the API review process b/191371676
673 var flag string
Colin Cross0c19b692024-11-06 13:50:21 -0800674 if ctx.notInPlatform() {
Spandan Das357ffcc2024-07-24 18:07:48 +0000675 flag = "--apex"
676 } else {
677 flag = "--systemapi"
678 }
679 // b/184712170, unless the lib is an NDK library, exclude all public symbols from
680 // the stub so that it is mandated that all symbols are explicitly marked with
681 // either apex or systemapi.
682 if !ctx.Module().(*Module).IsNdk(ctx.Config()) &&
683 // the symbol files of libclang libs are autogenerated and do not contain systemapi tags
684 // TODO (spandandas): Update mapfile.py to include #systemapi tag on all symbols
685 !strings.Contains(ctx.ModuleName(), "libclang_rt") {
686 flag = flag + " --no-ndk"
687 }
Spandan Dasc27dd332024-08-15 21:49:14 +0000688 // TODO(b/361303067): Remove this special case if bionic/ projects are added to ART development branches.
689 if isBionic(ctx.baseModuleName()) {
690 // set the flags explicitly for bionic libs.
691 // this is necessary for development in minimal branches which does not contain bionic/*.
692 // In such minimal branches, e.g. on the prebuilt libc stubs
693 // 1. IsNdk will return false (since the ndk_library definition for libc does not exist)
694 // 2. NotInPlatform will return true (since the source com.android.runtime does not exist)
695 flag = "--apex"
696 }
Spandan Das357ffcc2024-07-24 18:07:48 +0000697 nativeAbiResult := parseNativeAbiDefinition(ctx, symbolFile,
698 android.ApiLevelOrPanic(ctx, library.MutatedProperties.StubsVersion), flag)
699 objs := compileStubLibrary(ctx, flags, nativeAbiResult.stubSrc)
700
701 library.versionScriptPath = android.OptionalPathForPath(
702 nativeAbiResult.versionScript)
703
704 // Parse symbol file to get API list for coverage
705 if library.stubsVersion() == "current" && ctx.PrimaryArch() && !ctx.inRecovery() && !ctx.inProduct() && !ctx.inVendor() {
706 library.apiListCoverageXmlPath = parseSymbolFileForAPICoverage(ctx, symbolFile)
707 }
708
709 return objs
710}
711
Colin Crossb916a382016-07-29 17:28:03 -0700712type libraryInterface interface {
Colin Cross3572cf72020-10-01 15:58:11 -0700713 versionedInterface
714
Colin Crossb916a382016-07-29 17:28:03 -0700715 static() bool
Inseob Kimae553032019-05-14 18:52:49 +0900716 shared() bool
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700717 objs() Objects
Colin Cross0de8a1e2020-09-18 14:15:30 -0700718 reuseObjs() Objects
Colin Cross26c34ed2016-09-30 17:10:16 -0700719 toc() android.OptionalPath
Colin Crossb916a382016-07-29 17:28:03 -0700720
721 // Returns true if the build options for the module have selected a static or shared build
722 buildStatic() bool
723 buildShared() bool
724
725 // Sets whether a specific variant is static or shared
Colin Crossa48ab5b2017-02-14 15:28:44 -0800726 setStatic()
727 setShared()
Logan Chien41eabe62019-04-10 13:33:58 +0800728
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +0800729 // Gets the ABI properties for vendor, product, or platform variant
Colin Cross91ae5ec2024-10-01 14:03:40 -0700730 getHeaderAbiCheckerProperties(m *Module) headerAbiCheckerProperties
Yo Chiang2bbadfa2020-12-14 11:42:16 +0800731
Logan Chien41eabe62019-04-10 13:33:58 +0800732 // Write LOCAL_ADDITIONAL_DEPENDENCIES for ABI diff
733 androidMkWriteAdditionalDependenciesForSourceAbiDiff(w io.Writer)
Jiyong Parka90ca002019-10-07 15:47:24 +0900734
Yu Liub73c3a62024-12-10 00:58:06 +0000735 apexAvailable() []string
Colin Crossceaa5322021-09-28 16:37:50 -0700736
737 getAPIListCoverageXMLPath() android.ModuleOutPath
Colin Cross1bc94122021-10-28 13:25:54 -0700738
739 installable() *bool
Colin Crossb916a382016-07-29 17:28:03 -0700740}
741
Colin Crossc88c2722020-09-28 17:32:47 -0700742type versionedInterface interface {
743 buildStubs() bool
Jiyong Parkd4a3a132021-03-17 20:21:35 +0900744 setBuildStubs(isLatest bool)
Colin Crossc88c2722020-09-28 17:32:47 -0700745 hasStubsVariants() bool
Alan Stokes73feba32022-11-14 12:21:24 +0000746 isStubsImplementationRequired() bool
Colin Crossc88c2722020-09-28 17:32:47 -0700747 setStubsVersion(string)
748 stubsVersion() string
749
Colin Crossadd04a82024-05-22 09:57:59 -0700750 stubsVersions(ctx android.BaseModuleContext) []string
Colin Crossc88c2722020-09-28 17:32:47 -0700751 setAllStubsVersions([]string)
752 allStubsVersions() []string
Colin Cross0477b422020-10-13 18:43:54 -0700753
754 implementationModuleName(name string) string
Colin Cross127bb8b2020-12-16 16:46:01 -0800755 hasLLNDKStubs() bool
Colin Cross1f3f1302021-04-26 18:37:44 -0700756 hasLLNDKHeaders() bool
Colin Cross5271fea2021-04-27 13:06:04 -0700757 hasVendorPublicLibrary() bool
Colin Cross0510f542024-11-13 16:07:08 -0800758 isLLNDKMovedToApex() bool
Colin Crossc88c2722020-09-28 17:32:47 -0700759}
760
761var _ libraryInterface = (*libraryDecorator)(nil)
762var _ versionedInterface = (*libraryDecorator)(nil)
763
Justin Yun6977e8a2020-10-29 18:24:11 +0900764func (library *libraryDecorator) getLibNameHelper(baseModuleName string, inVendor bool, inProduct bool) string {
Colin Crossb916a382016-07-29 17:28:03 -0700765 name := library.libName
766 if name == "" {
dimitryd95964a2018-11-07 13:43:34 +0100767 name = String(library.Properties.Stem)
768 if name == "" {
Jooyung Han0302a842019-10-30 18:43:49 +0900769 name = baseModuleName
dimitryd95964a2018-11-07 13:43:34 +0100770 }
Colin Crossb916a382016-07-29 17:28:03 -0700771 }
772
Colin Cross0fd6a412019-08-16 14:22:10 -0700773 suffix := ""
Justin Yun6977e8a2020-10-29 18:24:11 +0900774 if inVendor {
Colin Cross0fd6a412019-08-16 14:22:10 -0700775 suffix = String(library.Properties.Target.Vendor.Suffix)
Justin Yun6977e8a2020-10-29 18:24:11 +0900776 } else if inProduct {
777 suffix = String(library.Properties.Target.Product.Suffix)
Colin Cross0fd6a412019-08-16 14:22:10 -0700778 }
779 if suffix == "" {
780 suffix = String(library.Properties.Suffix)
781 }
782
Jooyung Han0302a842019-10-30 18:43:49 +0900783 return name + suffix
784}
785
Chris Parsons3c27ca32020-11-20 12:42:07 -0500786// getLibName returns the actual canonical name of the library (the name which
787// should be passed to the linker via linker flags).
Jooyung Han0302a842019-10-30 18:43:49 +0900788func (library *libraryDecorator) getLibName(ctx BaseModuleContext) string {
Justin Yun6977e8a2020-10-29 18:24:11 +0900789 name := library.getLibNameHelper(ctx.baseModuleName(), ctx.inVendor(), ctx.inProduct())
Colin Cross0fd6a412019-08-16 14:22:10 -0700790
Colin Crossb916a382016-07-29 17:28:03 -0700791 if ctx.Host() && Bool(library.Properties.Unique_host_soname) {
792 if !strings.HasSuffix(name, "-host") {
793 name = name + "-host"
794 }
795 }
796
Inseob Kim0ce291e2019-07-04 14:38:27 +0900797 return name
Colin Crossb916a382016-07-29 17:28:03 -0700798}
799
Jiyong Parkda732bd2018-11-02 18:23:15 +0900800var versioningMacroNamesListMutex sync.Mutex
801
Colin Crossb916a382016-07-29 17:28:03 -0700802func (library *libraryDecorator) linkerInit(ctx BaseModuleContext) {
803 location := InstallInSystem
Dan Albert61f32122018-07-26 14:00:24 -0700804 if library.baseLinker.sanitize.inSanitizerDir() {
Vishwath Mohan1dd88392017-03-29 22:00:18 -0700805 location = InstallInSanitizerDir
Colin Crossb916a382016-07-29 17:28:03 -0700806 }
807 library.baseInstaller.location = location
Colin Crossb916a382016-07-29 17:28:03 -0700808 library.baseLinker.linkerInit(ctx)
Jiyong Park7ed9de32018-10-15 22:25:07 +0900809 // Let baseLinker know whether this variant is for stubs or not, so that
810 // it can omit things that are not required for linking stubs.
811 library.baseLinker.dynamicProperties.BuildStubs = library.buildStubs()
Jiyong Parkda732bd2018-11-02 18:23:15 +0900812
813 if library.buildStubs() {
814 macroNames := versioningMacroNamesList(ctx.Config())
815 myName := versioningMacroName(ctx.ModuleName())
816 versioningMacroNamesListMutex.Lock()
817 defer versioningMacroNamesListMutex.Unlock()
818 if (*macroNames)[myName] == "" {
819 (*macroNames)[myName] = ctx.ModuleName()
820 } else if (*macroNames)[myName] != ctx.ModuleName() {
821 ctx.ModuleErrorf("Macro name %q for versioning conflicts with macro name from module %q ", myName, (*macroNames)[myName])
822 }
823 }
Colin Crossb916a382016-07-29 17:28:03 -0700824}
825
dimitry0345ad82018-12-05 16:28:14 +0100826func (library *libraryDecorator) compilerDeps(ctx DepsContext, deps Deps) Deps {
Colin Cross127bb8b2020-12-16 16:46:01 -0800827 if ctx.IsLlndk() {
828 // LLNDK libraries ignore most of the properties on the cc_library and use the
829 // LLNDK-specific properties instead.
830 return deps
831 }
832
dimitry0345ad82018-12-05 16:28:14 +0100833 deps = library.baseCompiler.compilerDeps(ctx, deps)
834
dimitry0345ad82018-12-05 16:28:14 +0100835 return deps
836}
837
Colin Cross37047f12016-12-13 17:06:13 -0800838func (library *libraryDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
Colin Cross127bb8b2020-12-16 16:46:01 -0800839 if ctx.IsLlndk() {
840 // LLNDK libraries ignore most of the properties on the cc_library and use the
841 // LLNDK-specific properties instead.
Colin Cross0fb7fcd2021-03-02 11:00:07 -0800842 deps.HeaderLibs = append([]string(nil), library.Properties.Llndk.Export_llndk_headers...)
843 deps.ReexportHeaderLibHeaders = append([]string(nil), library.Properties.Llndk.Export_llndk_headers...)
Colin Cross127bb8b2020-12-16 16:46:01 -0800844 return deps
845 }
Colin Cross5271fea2021-04-27 13:06:04 -0700846 if ctx.IsVendorPublicLibrary() {
847 headers := library.Properties.Vendor_public_library.Export_public_headers
848 deps.HeaderLibs = append([]string(nil), headers...)
849 deps.ReexportHeaderLibHeaders = append([]string(nil), headers...)
850 return deps
851 }
Colin Cross127bb8b2020-12-16 16:46:01 -0800852
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800853 if library.static() {
Martin Stjernholm10566a02020-03-24 01:19:52 +0000854 // Compare with nil because an empty list needs to be propagated.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700855 if library.StaticProperties.Static.System_shared_libs != nil {
856 library.baseLinker.Properties.System_shared_libs = library.StaticProperties.Static.System_shared_libs
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800857 }
858 } else if library.shared() {
Martin Stjernholm10566a02020-03-24 01:19:52 +0000859 // Compare with nil because an empty list needs to be propagated.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700860 if library.SharedProperties.Shared.System_shared_libs != nil {
861 library.baseLinker.Properties.System_shared_libs = library.SharedProperties.Shared.System_shared_libs
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800862 }
863 }
864
Colin Crossb916a382016-07-29 17:28:03 -0700865 deps = library.baseLinker.linkerDeps(ctx, deps)
866
867 if library.static() {
868 deps.WholeStaticLibs = append(deps.WholeStaticLibs,
Cole Faustf0006e72024-08-19 14:39:19 -0700869 library.StaticProperties.Static.Whole_static_libs.GetOrDefault(ctx, nil)...)
870 deps.StaticLibs = append(deps.StaticLibs, library.StaticProperties.Static.Static_libs.GetOrDefault(ctx, nil)...)
871 deps.SharedLibs = append(deps.SharedLibs, library.StaticProperties.Static.Shared_libs.GetOrDefault(ctx, nil)...)
Colin Crosseefe9a32019-01-22 14:41:08 -0800872
Colin Crosse1bb5d02019-09-24 14:55:04 -0700873 deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, library.StaticProperties.Static.Export_shared_lib_headers...)
874 deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, library.StaticProperties.Static.Export_static_lib_headers...)
Colin Crossa48ab5b2017-02-14 15:28:44 -0800875 } else if library.shared() {
Alex Márquez Pérez Muñíz Díaz Puras Thaureaux01ec55e2023-01-30 22:53:04 +0000876 if library.baseLinker.Properties.crt() {
Colin Crossd1a28132021-06-21 17:34:47 -0700877 deps.CrtBegin = append(deps.CrtBegin, ctx.toolchain().CrtBeginSharedLibrary()...)
878 deps.CrtEnd = append(deps.CrtEnd, ctx.toolchain().CrtEndSharedLibrary()...)
Kalesh Singhf4ffe0a2024-01-29 13:01:51 -0800879
880 }
881 if library.baseLinker.Properties.crtPadSegment() {
882 deps.CrtEnd = append(deps.CrtEnd, ctx.toolchain().CrtPadSegmentSharedLibrary()...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700883 }
Cole Faustf0006e72024-08-19 14:39:19 -0700884 deps.WholeStaticLibs = append(deps.WholeStaticLibs, library.SharedProperties.Shared.Whole_static_libs.GetOrDefault(ctx, nil)...)
885 deps.StaticLibs = append(deps.StaticLibs, library.SharedProperties.Shared.Static_libs.GetOrDefault(ctx, nil)...)
886 deps.SharedLibs = append(deps.SharedLibs, library.SharedProperties.Shared.Shared_libs.GetOrDefault(ctx, nil)...)
Colin Crosseefe9a32019-01-22 14:41:08 -0800887
Colin Crosse1bb5d02019-09-24 14:55:04 -0700888 deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, library.SharedProperties.Shared.Export_shared_lib_headers...)
889 deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, library.SharedProperties.Shared.Export_static_lib_headers...)
Hsin-Yi Chen715142a2024-03-27 16:31:16 +0800890
891 deps.LlndkHeaderLibs = append(deps.LlndkHeaderLibs, library.Properties.Llndk.Export_llndk_headers...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700892 }
Justin Yun6977e8a2020-10-29 18:24:11 +0900893 if ctx.inVendor() {
Jiyong Park52d25bd2017-10-13 09:17:01 +0900894 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
895 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Vendor.Exclude_shared_libs)
896 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
Victor Chang51271c12019-01-30 16:02:22 +0000897 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Vendor.Exclude_shared_libs)
898 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
Jiyong Park52d25bd2017-10-13 09:17:01 +0900899 }
Justin Yun6977e8a2020-10-29 18:24:11 +0900900 if ctx.inProduct() {
901 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
902 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Product.Exclude_shared_libs)
903 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
904 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Product.Exclude_shared_libs)
905 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
906 }
Jiyong Parkf9332f12018-02-01 00:54:12 +0900907 if ctx.inRecovery() {
908 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
909 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Recovery.Exclude_shared_libs)
910 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
Victor Chang51271c12019-01-30 16:02:22 +0000911 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Recovery.Exclude_shared_libs)
912 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
Jiyong Parkf9332f12018-02-01 00:54:12 +0900913 }
Yifan Hongcf4832c2020-01-21 17:04:13 -0800914 if ctx.inRamdisk() {
915 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
916 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_shared_libs)
917 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
918 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Ramdisk.Exclude_shared_libs)
919 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
920 }
Yifan Hong6da33c22020-10-27 15:01:21 -0700921 if ctx.inVendorRamdisk() {
922 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
923 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
924 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
925 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
926 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
927 }
Colin Cross2383f3b2018-02-06 14:40:13 -0800928
Colin Cross4d9c2d12016-07-29 12:48:20 -0700929 return deps
930}
931
Cole Fauste8a87832024-09-11 11:35:46 -0700932func (library *libraryDecorator) linkerSpecifiedDeps(ctx android.ConfigurableEvaluatorContext, module *Module, specifiedDeps specifiedDeps) specifiedDeps {
Cole Faustf0006e72024-08-19 14:39:19 -0700933 specifiedDeps = library.baseLinker.linkerSpecifiedDeps(ctx, module, specifiedDeps)
Paul Duffin13f02712020-03-06 12:30:43 +0000934 var properties StaticOrSharedProperties
935 if library.static() {
936 properties = library.StaticProperties.Static
937 } else if library.shared() {
938 properties = library.SharedProperties.Shared
939 }
940
Cole Faustf0006e72024-08-19 14:39:19 -0700941 eval := module.ConfigurableEvaluator(ctx)
942 specifiedDeps.sharedLibs = append(specifiedDeps.sharedLibs, properties.Shared_libs.GetOrDefault(eval, nil)...)
Martin Stjernholm10566a02020-03-24 01:19:52 +0000943
944 // Must distinguish nil and [] in system_shared_libs - ensure that [] in
945 // either input list doesn't come out as nil.
946 if specifiedDeps.systemSharedLibs == nil {
947 specifiedDeps.systemSharedLibs = properties.System_shared_libs
948 } else {
949 specifiedDeps.systemSharedLibs = append(specifiedDeps.systemSharedLibs, properties.System_shared_libs...)
950 }
Paul Duffin13f02712020-03-06 12:30:43 +0000951
952 specifiedDeps.sharedLibs = android.FirstUniqueStrings(specifiedDeps.sharedLibs)
Martin Stjernholm10566a02020-03-24 01:19:52 +0000953 if len(specifiedDeps.systemSharedLibs) > 0 {
954 // Skip this if systemSharedLibs is either nil or [], to ensure they are
955 // retained.
956 specifiedDeps.systemSharedLibs = android.FirstUniqueStrings(specifiedDeps.systemSharedLibs)
957 }
Paul Duffin13f02712020-03-06 12:30:43 +0000958 return specifiedDeps
959}
960
Colin Cross4a9e6ec2023-12-18 15:29:41 -0800961func (library *libraryDecorator) moduleInfoJSON(ctx ModuleContext, moduleInfoJSON *android.ModuleInfoJSON) {
962 if library.static() {
963 moduleInfoJSON.Class = []string{"STATIC_LIBRARIES"}
964 moduleInfoJSON.Uninstallable = true
965 } else if library.shared() {
966 moduleInfoJSON.Class = []string{"SHARED_LIBRARIES"}
967 } else if library.header() {
968 moduleInfoJSON.Class = []string{"HEADER_LIBRARIES"}
969 moduleInfoJSON.Uninstallable = true
970 }
971
972 if library.buildStubs() && library.stubsVersion() != "" {
973 moduleInfoJSON.SubName += "." + library.stubsVersion()
974 }
975
976 // If a library providing a stub is included in an APEX, the private APIs of the library
977 // is accessible only inside the APEX. From outside of the APEX, clients can only use the
978 // public APIs via the stub. To enforce this, the (latest version of the) stub gets the
979 // name of the library. The impl library instead gets the `.bootstrap` suffix to so that
980 // they can be exceptionally used directly when APEXes are not available (e.g. during the
981 // very early stage in the boot process).
982 if len(library.Properties.Stubs.Versions) > 0 && !ctx.Host() && ctx.notInPlatform() &&
983 !ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() && !ctx.useVndk() && !ctx.static() {
984 if library.buildStubs() && library.isLatestStubVersion() {
985 moduleInfoJSON.SubName = ""
986 }
987 if !library.buildStubs() {
988 moduleInfoJSON.SubName = ".bootstrap"
989 }
990 }
991
992 library.baseLinker.moduleInfoJSON(ctx, moduleInfoJSON)
993}
994
Colin Crossb916a382016-07-29 17:28:03 -0700995func (library *libraryDecorator) linkStatic(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700996 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700997
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700998 library.objects = deps.WholeStaticLibObjs.Copy()
999 library.objects = library.objects.Append(objs)
Colin Crossa2bcf2c2022-02-11 13:11:55 -08001000 library.wholeStaticLibsFromPrebuilts = android.CopyOfPaths(deps.WholeStaticLibsFromPrebuilts)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001001
Inseob Kim0ce291e2019-07-04 14:38:27 +09001002 fileName := ctx.ModuleName() + staticLibraryExtension
Colin Cross86803cf2018-02-15 14:12:26 -08001003 outputFile := android.PathForModuleOut(ctx, fileName)
Dan Willemsen581341d2017-02-09 16:16:31 -08001004 builderFlags := flagsToBuilderFlags(flags)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001005
Dan Willemsen569edc52018-11-19 09:33:29 -08001006 if Bool(library.baseLinker.Properties.Use_version_lib) {
1007 if ctx.Host() {
1008 versionedOutputFile := outputFile
1009 outputFile = android.PathForModuleOut(ctx, "unversioned", fileName)
1010 library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
1011 } else {
1012 versionedOutputFile := android.PathForModuleOut(ctx, "versioned", fileName)
Jingwen Chen40fd90a2020-06-15 05:24:19 +00001013 library.distFile = versionedOutputFile
Dan Willemsen569edc52018-11-19 09:33:29 -08001014 library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
1015 }
Colin Cross86803cf2018-02-15 14:12:26 -08001016 }
1017
Chih-Hung Hsieh7540a782022-01-08 19:56:09 -08001018 transformObjToStaticLib(ctx, library.objects.objFiles, deps.WholeStaticLibsFromPrebuilts, builderFlags, outputFile, nil, objs.tidyDepFiles)
Dan Willemsen581341d2017-02-09 16:16:31 -08001019
Chris Parsonsbf4f55f2020-11-23 17:02:44 -05001020 library.coverageOutputFile = transformCoverageFilesToZip(ctx, library.objects, ctx.ModuleName())
Colin Cross4d9c2d12016-07-29 12:48:20 -07001021
Colin Cross4d9c2d12016-07-29 12:48:20 -07001022 ctx.CheckbuildFile(outputFile)
1023
Colin Cross649d8172020-12-10 12:30:21 -08001024 if library.static() {
Colin Cross40213022023-12-13 15:19:49 -08001025 android.SetProvider(ctx, StaticLibraryInfoProvider, StaticLibraryInfo{
Colin Crossa2bcf2c2022-02-11 13:11:55 -08001026 StaticLibrary: outputFile,
1027 ReuseObjects: library.reuseObjects,
1028 Objects: library.objects,
1029 WholeStaticLibsFromPrebuilts: library.wholeStaticLibsFromPrebuilts,
Colin Cross0de8a1e2020-09-18 14:15:30 -07001030
Colin Crossa14fb6a2024-10-23 16:57:06 -07001031 TransitiveStaticLibrariesForOrdering: depset.NewBuilder[android.Path](depset.TOPOLOGICAL).
Colin Cross649d8172020-12-10 12:30:21 -08001032 Direct(outputFile).
1033 Transitive(deps.TranstiveStaticLibrariesForOrdering).
1034 Build(),
1035 })
1036 }
1037
1038 if library.header() {
Colin Cross40213022023-12-13 15:19:49 -08001039 android.SetProvider(ctx, HeaderLibraryInfoProvider, HeaderLibraryInfo{})
Colin Cross649d8172020-12-10 12:30:21 -08001040 }
Colin Cross0de8a1e2020-09-18 14:15:30 -07001041
Colin Cross4d9c2d12016-07-29 12:48:20 -07001042 return outputFile
1043}
1044
Chih-Hung Hsiehf6ca1b92021-12-05 18:02:50 -08001045func ndkSharedLibDeps(ctx ModuleContext) android.Paths {
1046 if ctx.Module().(*Module).IsSdkVariant() {
1047 // The NDK sysroot timestamp file depends on all the NDK
1048 // sysroot header and shared library files.
1049 return android.Paths{getNdkBaseTimestampFile(ctx)}
1050 }
1051 return nil
1052}
1053
Colin Crossb916a382016-07-29 17:28:03 -07001054func (library *libraryDecorator) linkShared(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001055 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Cross4d9c2d12016-07-29 12:48:20 -07001056
1057 var linkerDeps android.Paths
Pirama Arumuga Nainarada83ec2017-08-31 23:38:27 -07001058 linkerDeps = append(linkerDeps, flags.LdFlagsDeps...)
Chih-Hung Hsiehf6ca1b92021-12-05 18:02:50 -08001059 linkerDeps = append(linkerDeps, ndkSharedLibDeps(ctx)...)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001060
Jerome Gaillard9b454732024-12-10 18:07:19 +00001061 exportedSymbols := ctx.ExpandOptionalSource(library.Properties.Exported_symbols_list, "exported_symbols_list")
Colin Cross2383f3b2018-02-06 14:40:13 -08001062 unexportedSymbols := ctx.ExpandOptionalSource(library.Properties.Unexported_symbols_list, "unexported_symbols_list")
1063 forceNotWeakSymbols := ctx.ExpandOptionalSource(library.Properties.Force_symbols_not_weak_list, "force_symbols_not_weak_list")
1064 forceWeakSymbols := ctx.ExpandOptionalSource(library.Properties.Force_symbols_weak_list, "force_symbols_weak_list")
Colin Cross4d9c2d12016-07-29 12:48:20 -07001065 if !ctx.Darwin() {
Jerome Gaillard9b454732024-12-10 18:07:19 +00001066 if exportedSymbols.Valid() {
1067 ctx.PropertyErrorf("exported_symbols_list", "Only supported on Darwin")
1068 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001069 if unexportedSymbols.Valid() {
1070 ctx.PropertyErrorf("unexported_symbols_list", "Only supported on Darwin")
1071 }
1072 if forceNotWeakSymbols.Valid() {
1073 ctx.PropertyErrorf("force_symbols_not_weak_list", "Only supported on Darwin")
1074 }
1075 if forceWeakSymbols.Valid() {
1076 ctx.PropertyErrorf("force_symbols_weak_list", "Only supported on Darwin")
1077 }
1078 } else {
Jerome Gaillard9b454732024-12-10 18:07:19 +00001079 if exportedSymbols.Valid() {
1080 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-exported_symbols_list,"+exportedSymbols.String())
1081 linkerDeps = append(linkerDeps, exportedSymbols.Path())
1082 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001083 if unexportedSymbols.Valid() {
Colin Cross4af21ed2019-11-04 09:37:55 -08001084 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-unexported_symbols_list,"+unexportedSymbols.String())
Colin Cross4d9c2d12016-07-29 12:48:20 -07001085 linkerDeps = append(linkerDeps, unexportedSymbols.Path())
1086 }
1087 if forceNotWeakSymbols.Valid() {
Colin Cross4af21ed2019-11-04 09:37:55 -08001088 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-force_symbols_not_weak_list,"+forceNotWeakSymbols.String())
Colin Cross4d9c2d12016-07-29 12:48:20 -07001089 linkerDeps = append(linkerDeps, forceNotWeakSymbols.Path())
1090 }
1091 if forceWeakSymbols.Valid() {
Colin Cross4af21ed2019-11-04 09:37:55 -08001092 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-force_symbols_weak_list,"+forceWeakSymbols.String())
Colin Cross4d9c2d12016-07-29 12:48:20 -07001093 linkerDeps = append(linkerDeps, forceWeakSymbols.Path())
1094 }
1095 }
Colin Cross8e21aa52020-09-28 18:28:02 -07001096 if library.versionScriptPath.Valid() {
Jiyong Parkc1e7f482019-01-12 13:39:10 +09001097 linkerScriptFlags := "-Wl,--version-script," + library.versionScriptPath.String()
Colin Cross4af21ed2019-11-04 09:37:55 -08001098 flags.Local.LdFlags = append(flags.Local.LdFlags, linkerScriptFlags)
Colin Cross8e21aa52020-09-28 18:28:02 -07001099 linkerDeps = append(linkerDeps, library.versionScriptPath.Path())
Jiyong Parkc1e7f482019-01-12 13:39:10 +09001100 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001101
1102 fileName := library.getLibName(ctx) + flags.Toolchain.ShlibSuffix()
1103 outputFile := android.PathForModuleOut(ctx, fileName)
Colin Cross0de8a1e2020-09-18 14:15:30 -07001104 unstrippedOutputFile := outputFile
Colin Cross4d9c2d12016-07-29 12:48:20 -07001105
Josh Gao75a50a22019-06-07 17:58:59 -07001106 var implicitOutputs android.WritablePaths
1107 if ctx.Windows() {
1108 importLibraryPath := android.PathForModuleOut(ctx, pathtools.ReplaceExtension(fileName, "lib"))
1109
Colin Cross4af21ed2019-11-04 09:37:55 -08001110 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,--out-implib="+importLibraryPath.String())
Josh Gao75a50a22019-06-07 17:58:59 -07001111 implicitOutputs = append(implicitOutputs, importLibraryPath)
1112 }
1113
Colin Cross4d9c2d12016-07-29 12:48:20 -07001114 builderFlags := flagsToBuilderFlags(flags)
1115
Dan Willemsen47450072021-10-19 20:24:49 -07001116 if ctx.Darwin() && deps.DarwinSecondArchOutput.Valid() {
1117 fatOutputFile := outputFile
1118 outputFile = android.PathForModuleOut(ctx, "pre-fat", fileName)
1119 transformDarwinUniversalBinary(ctx, fatOutputFile, outputFile, deps.DarwinSecondArchOutput.Path())
1120 }
1121
Colin Crossb496cfd2018-09-10 16:50:05 -07001122 // Optimize out relinking against shared libraries whose interface hasn't changed by
1123 // depending on a table of contents file instead of the library itself.
Colin Cross70dda7e2019-10-01 22:05:35 -07001124 tocFile := outputFile.ReplaceExtension(ctx, flags.Toolchain.ShlibSuffix()[1:]+".toc")
Colin Crossb496cfd2018-09-10 16:50:05 -07001125 library.tocFile = android.OptionalPathForPath(tocFile)
Ivan Lozano7b0781d2021-11-03 15:30:18 -04001126 TransformSharedObjectToToc(ctx, outputFile, tocFile)
Colin Cross89562dc2016-10-03 17:47:19 -07001127
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001128 stripFlags := flagsToStripFlags(flags)
Colin Crossadc81a02020-12-14 17:01:55 -08001129 needsStrip := library.stripper.NeedsStrip(ctx)
1130 if library.buildStubs() {
1131 // No need to strip stubs libraries
1132 needsStrip = false
1133 }
1134 if needsStrip {
Yi Kongb5c34d72018-11-07 16:28:49 -08001135 if ctx.Darwin() {
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001136 stripFlags.StripUseGnuStrip = true
Yi Kongb5c34d72018-11-07 16:28:49 -08001137 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001138 strippedOutputFile := outputFile
1139 outputFile = android.PathForModuleOut(ctx, "unstripped", fileName)
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001140 library.stripper.StripExecutableOrSharedLib(ctx, outputFile, strippedOutputFile, stripFlags)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001141 }
Colin Crossb60190a2018-09-04 16:28:17 -07001142 library.unstrippedOutputFile = outputFile
1143
Colin Crossd7227f92019-09-05 14:26:33 -07001144 outputFile = maybeInjectBoringSSLHash(ctx, outputFile, library.Properties.Inject_bssl_hash, fileName)
Pete Bentley803e1612019-08-06 22:19:59 +01001145
Dan Willemsen569edc52018-11-19 09:33:29 -08001146 if Bool(library.baseLinker.Properties.Use_version_lib) {
1147 if ctx.Host() {
1148 versionedOutputFile := outputFile
1149 outputFile = android.PathForModuleOut(ctx, "unversioned", fileName)
1150 library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
1151 } else {
1152 versionedOutputFile := android.PathForModuleOut(ctx, "versioned", fileName)
Jingwen Chen40fd90a2020-06-15 05:24:19 +00001153 library.distFile = versionedOutputFile
Dan Willemsen569edc52018-11-19 09:33:29 -08001154
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001155 if library.stripper.NeedsStrip(ctx) {
Dan Willemsen569edc52018-11-19 09:33:29 -08001156 out := android.PathForModuleOut(ctx, "versioned-stripped", fileName)
Jingwen Chen40fd90a2020-06-15 05:24:19 +00001157 library.distFile = out
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001158 library.stripper.StripExecutableOrSharedLib(ctx, versionedOutputFile, out, stripFlags)
Dan Willemsen569edc52018-11-19 09:33:29 -08001159 }
1160
1161 library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
1162 }
Colin Cross86803cf2018-02-15 14:12:26 -08001163 }
1164
Wei Li5f5d2712023-12-11 15:40:29 -08001165 // Generate an output file for dist as if strip: "all" is set on the module.
1166 // Currently this is for layoutlib release process only.
1167 for _, dist := range ctx.Module().(*Module).Dists() {
1168 if dist.Tag != nil && *dist.Tag == "stripped_all" {
1169 strippedAllOutputFile := android.PathForModuleOut(ctx, "stripped_all", fileName)
1170 transformStrip(ctx, outputFile, strippedAllOutputFile, StripFlags{Toolchain: flags.Toolchain})
1171 library.strippedAllOutputFile = strippedAllOutputFile
1172 break
1173 }
1174 }
1175
Jiyong Park64a44f22019-01-18 14:37:08 +09001176 sharedLibs := deps.EarlySharedLibs
1177 sharedLibs = append(sharedLibs, deps.SharedLibs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001178 sharedLibs = append(sharedLibs, deps.LateSharedLibs...)
1179
Jiyong Park64a44f22019-01-18 14:37:08 +09001180 linkerDeps = append(linkerDeps, deps.EarlySharedLibsDeps...)
Colin Cross26c34ed2016-09-30 17:10:16 -07001181 linkerDeps = append(linkerDeps, deps.SharedLibsDeps...)
1182 linkerDeps = append(linkerDeps, deps.LateSharedLibsDeps...)
Ivan Lozano0a468a42024-05-13 21:03:34 -04001183
Ivan Lozanofd47b1a2024-05-17 14:13:41 -04001184 if generatedLib := generateRustStaticlib(ctx, deps.RustRlibDeps); generatedLib != nil && !library.buildStubs() {
1185 if ctx.Module().(*Module).WholeRustStaticlib {
1186 deps.WholeStaticLibs = append(deps.WholeStaticLibs, generatedLib)
1187 } else {
1188 deps.StaticLibs = append(deps.StaticLibs, generatedLib)
1189 }
Ivan Lozano0a468a42024-05-13 21:03:34 -04001190 }
1191
Chris Parsonsbf4f55f2020-11-23 17:02:44 -05001192 transformObjToDynamicBinary(ctx, objs.objFiles, sharedLibs,
Ivan Lozano0a468a42024-05-13 21:03:34 -04001193 deps.StaticLibs, deps.LateStaticLibs, deps.WholeStaticLibs, linkerDeps, deps.CrtBegin,
1194 deps.CrtEnd, false, builderFlags, outputFile, implicitOutputs, objs.tidyDepFiles)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001195
Dan Willemsen581341d2017-02-09 16:16:31 -08001196 objs.coverageFiles = append(objs.coverageFiles, deps.StaticLibObjs.coverageFiles...)
1197 objs.coverageFiles = append(objs.coverageFiles, deps.WholeStaticLibObjs.coverageFiles...)
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -08001198 objs.sAbiDumpFiles = append(objs.sAbiDumpFiles, deps.StaticLibObjs.sAbiDumpFiles...)
1199 objs.sAbiDumpFiles = append(objs.sAbiDumpFiles, deps.WholeStaticLibObjs.sAbiDumpFiles...)
1200
Chris Parsonsbf4f55f2020-11-23 17:02:44 -05001201 library.coverageOutputFile = transformCoverageFilesToZip(ctx, objs, library.getLibName(ctx))
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001202 library.linkSAbiDumpFiles(ctx, deps, objs, fileName, unstrippedOutputFile)
Dan Willemsen581341d2017-02-09 16:16:31 -08001203
Colin Crossa14fb6a2024-10-23 16:57:06 -07001204 var transitiveStaticLibrariesForOrdering depset.DepSet[android.Path]
Colin Cross0de8a1e2020-09-18 14:15:30 -07001205 if static := ctx.GetDirectDepsWithTag(staticVariantTag); len(static) > 0 {
Colin Cross313aa542023-12-13 13:47:44 -08001206 s, _ := android.OtherModuleProvider(ctx, static[0], StaticLibraryInfoProvider)
Liz Kammeref6dfea2021-06-08 15:37:09 -04001207 transitiveStaticLibrariesForOrdering = s.TransitiveStaticLibrariesForOrdering
Colin Cross0de8a1e2020-09-18 14:15:30 -07001208 }
1209
Colin Cross40213022023-12-13 15:19:49 -08001210 android.SetProvider(ctx, SharedLibraryInfoProvider, SharedLibraryInfo{
Liz Kammeref6dfea2021-06-08 15:37:09 -04001211 TableOfContents: android.OptionalPathForPath(tocFile),
1212 SharedLibrary: unstrippedOutputFile,
1213 TransitiveStaticLibrariesForOrdering: transitiveStaticLibrariesForOrdering,
1214 Target: ctx.Target(),
Colin Crossb614cd42024-10-11 12:52:21 -07001215 IsStubs: library.buildStubs(),
Colin Cross0de8a1e2020-09-18 14:15:30 -07001216 })
1217
Sam Delmerico75dbca22023-04-20 13:13:25 +00001218 addStubDependencyProviders(ctx)
1219
1220 return unstrippedOutputFile
1221}
1222
Spandan Das357ffcc2024-07-24 18:07:48 +00001223// Visits the stub variants of the library and returns a struct containing the stub .so paths
1224func addStubDependencyProviders(ctx ModuleContext) []SharedStubLibrary {
1225 stubsInfo := []SharedStubLibrary{}
Yu Liuffe86322024-12-18 18:53:12 +00001226 stubs := ctx.GetDirectDepsProxyWithTag(stubImplDepTag)
Colin Cross0de8a1e2020-09-18 14:15:30 -07001227 if len(stubs) > 0 {
Colin Cross0de8a1e2020-09-18 14:15:30 -07001228 for _, stub := range stubs {
Spandan Das357ffcc2024-07-24 18:07:48 +00001229 stubInfo, ok := android.OtherModuleProvider(ctx, stub, SharedLibraryInfoProvider)
1230 // TODO (b/275273834): Make this a hard error when the symbol files have been added to module sdk.
1231 if !ok {
1232 continue
1233 }
Colin Cross313aa542023-12-13 13:47:44 -08001234 flagInfo, _ := android.OtherModuleProvider(ctx, stub, FlagExporterInfoProvider)
Yu Liuffe86322024-12-18 18:53:12 +00001235 ccInfo, ok := android.OtherModuleProvider(ctx, stub, CcInfoProvider)
1236 if !ok || ccInfo.LibraryInfo == nil {
1237 panic(fmt.Errorf("couldn't find library info for %s", stub))
1238 }
Chris Parsons3c27ca32020-11-20 12:42:07 -05001239 stubsInfo = append(stubsInfo, SharedStubLibrary{
Yu Liuffe86322024-12-18 18:53:12 +00001240 Version: ccInfo.LibraryInfo.StubsVersion,
Colin Cross0de8a1e2020-09-18 14:15:30 -07001241 SharedLibraryInfo: stubInfo,
1242 FlagExporterInfo: flagInfo,
1243 })
1244 }
Spandan Das357ffcc2024-07-24 18:07:48 +00001245 if len(stubsInfo) > 0 {
1246 android.SetProvider(ctx, SharedLibraryStubsProvider, SharedLibraryStubsInfo{
1247 SharedStubLibraries: stubsInfo,
1248 IsLLNDK: ctx.IsLlndk(),
1249 })
1250 }
Colin Cross0de8a1e2020-09-18 14:15:30 -07001251 }
Spandan Das357ffcc2024-07-24 18:07:48 +00001252 return stubsInfo
Colin Cross4d9c2d12016-07-29 12:48:20 -07001253}
1254
Jiyong Parkaf6d8952019-01-31 12:21:23 +09001255func (library *libraryDecorator) unstrippedOutputFilePath() android.Path {
1256 return library.unstrippedOutputFile
1257}
1258
Wei Li5f5d2712023-12-11 15:40:29 -08001259func (library *libraryDecorator) strippedAllOutputFilePath() android.Path {
1260 return library.strippedAllOutputFile
1261}
1262
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001263func (library *libraryDecorator) disableStripping() {
1264 library.stripper.StripProperties.Strip.None = BoolPtr(true)
1265}
1266
Pirama Arumuga Nainar65c95ff2019-03-25 10:21:31 -07001267func (library *libraryDecorator) nativeCoverage() bool {
1268 if library.header() || library.buildStubs() {
1269 return false
1270 }
1271 return true
1272}
1273
Jiyong Parkee9a98d2019-08-09 14:44:36 +09001274func (library *libraryDecorator) coverageOutputFilePath() android.OptionalPath {
1275 return library.coverageOutputFile
1276}
1277
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001278func (library *libraryDecorator) exportedIncludeDirsForAbiCheck(ctx ModuleContext) []string {
1279 exportIncludeDirs := library.flagExporter.exportedIncludes(ctx).Strings()
1280 exportIncludeDirs = append(exportIncludeDirs, library.sabi.Properties.ReexportedIncludes...)
Hsin-Yi Chen5f228b02024-04-02 12:38:47 +08001281 exportSystemIncludeDirs := library.flagExporter.exportedSystemIncludes(ctx).Strings()
1282 exportSystemIncludeDirs = append(exportSystemIncludeDirs, library.sabi.Properties.ReexportedSystemIncludes...)
1283 // The ABI checker does not distinguish normal and system headers.
1284 return append(exportIncludeDirs, exportSystemIncludeDirs...)
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001285}
1286
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001287func (library *libraryDecorator) llndkIncludeDirsForAbiCheck(ctx ModuleContext, deps PathDeps) []string {
Hsin-Yi Chen5f228b02024-04-02 12:38:47 +08001288 var includeDirs, systemIncludeDirs []string
1289
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001290 if library.Properties.Llndk.Override_export_include_dirs != nil {
1291 includeDirs = append(includeDirs, android.PathsForModuleSrc(
1292 ctx, library.Properties.Llndk.Override_export_include_dirs).Strings()...)
1293 } else {
1294 includeDirs = append(includeDirs, library.flagExporter.exportedIncludes(ctx).Strings()...)
1295 // Ignore library.sabi.Properties.ReexportedIncludes because
1296 // LLNDK does not reexport the implementation's dependencies, such as export_header_libs.
1297 }
1298
Hsin-Yi Chen5f228b02024-04-02 12:38:47 +08001299 systemIncludeDirs = append(systemIncludeDirs,
1300 library.flagExporter.exportedSystemIncludes(ctx).Strings()...)
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001301 if Bool(library.Properties.Llndk.Export_headers_as_system) {
1302 systemIncludeDirs = append(systemIncludeDirs, includeDirs...)
1303 includeDirs = nil
1304 }
1305 // Header libs.
1306 includeDirs = append(includeDirs, deps.LlndkIncludeDirs.Strings()...)
1307 systemIncludeDirs = append(systemIncludeDirs, deps.LlndkSystemIncludeDirs.Strings()...)
1308 // The ABI checker does not distinguish normal and system headers.
1309 return append(includeDirs, systemIncludeDirs...)
1310}
1311
1312func (library *libraryDecorator) linkLlndkSAbiDumpFiles(ctx ModuleContext,
1313 deps PathDeps, sAbiDumpFiles android.Paths, soFile android.Path, libFileName string,
Hsin-Yi Chenf6fc5252024-04-15 15:43:21 +08001314 excludeSymbolVersions, excludeSymbolTags []string,
Hsin-Yi Chend6aaadc2024-11-13 16:58:46 +08001315 sdkVersionForVendorApiLevel string) android.Path {
Hsin-Yi Chenea748242025-01-15 15:44:48 +08001316 // Though LLNDK is implemented in system, the callers in vendor cannot include CommonGlobalIncludes,
1317 // so commonGlobalIncludes is false.
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001318 return transformDumpToLinkedDump(ctx,
1319 sAbiDumpFiles, soFile, libFileName+".llndk",
1320 library.llndkIncludeDirsForAbiCheck(ctx, deps),
1321 android.OptionalPathForModuleSrc(ctx, library.Properties.Llndk.Symbol_file),
1322 append([]string{"*_PLATFORM", "*_PRIVATE"}, excludeSymbolVersions...),
1323 append([]string{"platform-only"}, excludeSymbolTags...),
Hsin-Yi Chenea748242025-01-15 15:44:48 +08001324 []string{"llndk"}, sdkVersionForVendorApiLevel, false /* commonGlobalIncludes */)
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001325}
1326
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001327func (library *libraryDecorator) linkApexSAbiDumpFiles(ctx ModuleContext,
1328 deps PathDeps, sAbiDumpFiles android.Paths, soFile android.Path, libFileName string,
Hsin-Yi Chenf6fc5252024-04-15 15:43:21 +08001329 excludeSymbolVersions, excludeSymbolTags []string,
1330 sdkVersion string) android.Path {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001331 return transformDumpToLinkedDump(ctx,
1332 sAbiDumpFiles, soFile, libFileName+".apex",
1333 library.exportedIncludeDirsForAbiCheck(ctx),
1334 android.OptionalPathForModuleSrc(ctx, library.Properties.Stubs.Symbol_file),
1335 append([]string{"*_PLATFORM", "*_PRIVATE"}, excludeSymbolVersions...),
1336 append([]string{"platform-only"}, excludeSymbolTags...),
Hsin-Yi Chenea748242025-01-15 15:44:48 +08001337 []string{"apex", "systemapi"}, sdkVersion, requiresGlobalIncludes(ctx))
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001338}
1339
Hsin-Yi Chen0af4e672022-11-23 14:39:46 +08001340func getRefAbiDumpFile(ctx android.ModuleInstallPathContext,
1341 versionedDumpDir, fileName string) android.OptionalPath {
Mu-Le Leef5ed30b2022-08-30 10:37:21 +00001342
1343 currentArchType := ctx.Arch().ArchType
1344 primaryArchType := ctx.Config().DevicePrimaryArchType()
1345 archName := currentArchType.String()
1346 if currentArchType != primaryArchType {
1347 archName += "_" + primaryArchType.String()
1348 }
1349
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001350 return android.ExistentPathForSource(ctx, versionedDumpDir, archName, "source-based",
Hsin-Yi Chen0af4e672022-11-23 14:39:46 +08001351 fileName+".lsdump")
Mu-Le Leef5ed30b2022-08-30 10:37:21 +00001352}
1353
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001354// Return the previous and current SDK versions for cross-version ABI diff.
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001355func crossVersionAbiDiffSdkVersions(ctx ModuleContext, dumpDir string) (int, int) {
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001356 sdkVersionInt := ctx.Config().PlatformSdkVersion().FinalInt()
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001357
1358 if ctx.Config().PlatformSdkFinal() {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001359 return sdkVersionInt - 1, sdkVersionInt
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001360 } else {
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001361 // The platform SDK version can be upgraded before finalization while the corresponding abi dumps hasn't
1362 // been generated. Thus the Cross-Version Check chooses PLATFORM_SDK_VERION - 1 as previous version.
1363 // This situation could be identified by checking the existence of the PLATFORM_SDK_VERION dump directory.
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001364 versionedDumpDir := android.ExistentPathForSource(ctx,
1365 dumpDir, ctx.Config().PlatformSdkVersion().String())
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001366 if versionedDumpDir.Valid() {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001367 return sdkVersionInt, sdkVersionInt + 1
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001368 } else {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001369 return sdkVersionInt - 1, sdkVersionInt
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001370 }
1371 }
1372}
1373
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001374// Return the SDK version for same-version ABI diff.
1375func currRefAbiDumpSdkVersion(ctx ModuleContext) string {
Hsin-Yi Chen27bafd02024-01-08 18:38:42 +08001376 if ctx.Config().PlatformSdkFinal() {
Hsin-Yi Chena90bd382022-11-10 16:56:27 +08001377 // After sdk finalization, the ABI of the latest API level must be consistent with the source code,
1378 // so choose PLATFORM_SDK_VERSION as the current version.
1379 return ctx.Config().PlatformSdkVersion().String()
1380 } else {
1381 return "current"
1382 }
1383}
1384
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001385// sourceAbiDiff registers a build statement to compare linked sAbi dump files (.lsdump).
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001386func (library *libraryDecorator) sourceAbiDiff(ctx android.ModuleContext,
1387 sourceDump, referenceDump android.Path,
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001388 baseName, nameExt string, isLlndk, allowExtensions bool,
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001389 sourceVersion, errorMessage string) {
1390
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001391 extraFlags := []string{"-target-version", sourceVersion}
Colin Cross91ae5ec2024-10-01 14:03:40 -07001392 headerAbiChecker := library.getHeaderAbiCheckerProperties(ctx.Module().(*Module))
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +08001393 if Bool(headerAbiChecker.Check_all_apis) {
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001394 extraFlags = append(extraFlags, "-check-all-apis")
1395 } else {
1396 extraFlags = append(extraFlags,
1397 "-allow-unreferenced-changes",
1398 "-allow-unreferenced-elf-symbol-changes")
1399 }
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001400 if isLlndk {
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001401 extraFlags = append(extraFlags, "-consider-opaque-types-different")
1402 }
1403 if allowExtensions {
1404 extraFlags = append(extraFlags, "-allow-extensions")
1405 }
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +08001406 extraFlags = append(extraFlags, headerAbiChecker.Diff_flags...)
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001407
1408 library.sAbiDiff = append(
1409 library.sAbiDiff,
1410 transformAbiDumpToAbiDiff(ctx, sourceDump, referenceDump,
1411 baseName, nameExt, extraFlags, errorMessage))
1412}
1413
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001414func (library *libraryDecorator) crossVersionAbiDiff(ctx android.ModuleContext,
1415 sourceDump, referenceDump android.Path,
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001416 baseName, nameExt string, isLlndk bool, sourceVersion, prevDumpDir string) {
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001417
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001418 errorMessage := "error: Please follow https://android.googlesource.com/platform/development/+/main/vndk/tools/header-checker/README.md#configure-cross_version-abi-check to resolve the difference between your source code and the ABI dumps in " + prevDumpDir
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001419
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001420 library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, nameExt,
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001421 isLlndk, true /* allowExtensions */, sourceVersion, errorMessage)
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001422}
1423
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001424func (library *libraryDecorator) sameVersionAbiDiff(ctx android.ModuleContext,
1425 sourceDump, referenceDump android.Path,
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001426 baseName, nameExt string, isLlndk bool, lsdumpTagName string) {
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001427
1428 libName := strings.TrimSuffix(baseName, filepath.Ext(baseName))
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001429 errorMessage := "error: Please update ABI references with: $$ANDROID_BUILD_TOP/development/vndk/tools/header-checker/utils/create_reference_dumps.py --lib " + libName + " --lib-variant " + lsdumpTagName
1430
1431 targetRelease := ctx.Config().Getenv("TARGET_RELEASE")
1432 if targetRelease != "" {
1433 errorMessage += " --release " + targetRelease
1434 }
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001435
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001436 library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, nameExt,
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001437 isLlndk, false /* allowExtensions */, "current", errorMessage)
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001438}
1439
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001440func (library *libraryDecorator) optInAbiDiff(ctx android.ModuleContext,
1441 sourceDump, referenceDump android.Path,
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001442 baseName, nameExt string, refDumpDir string, lsdumpTagName string) {
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001443
1444 libName := strings.TrimSuffix(baseName, filepath.Ext(baseName))
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001445 errorMessage := "error: Please update ABI references with: $$ANDROID_BUILD_TOP/development/vndk/tools/header-checker/utils/create_reference_dumps.py --lib " + libName + " --lib-variant " + lsdumpTagName + " --ref-dump-dir $$ANDROID_BUILD_TOP/" + refDumpDir
1446
1447 targetRelease := ctx.Config().Getenv("TARGET_RELEASE")
1448 if targetRelease != "" {
1449 errorMessage += " --release " + targetRelease
1450 }
1451
Hsin-Yi Chen8b85d812023-06-29 16:51:27 +08001452 // Most opt-in libraries do not have dumps for all default architectures.
1453 if ctx.Config().HasDeviceProduct() {
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001454 errorMessage += " --product " + ctx.Config().DeviceProduct()
Hsin-Yi Chen8b85d812023-06-29 16:51:27 +08001455 }
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001456
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001457 library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, nameExt,
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001458 false /* isLlndk */, false /* allowExtensions */, "current", errorMessage)
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001459}
1460
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001461func (library *libraryDecorator) linkSAbiDumpFiles(ctx ModuleContext, deps PathDeps, objs Objects, fileName string, soFile android.Path) {
Yo Chiang2bbadfa2020-12-14 11:42:16 +08001462 if library.sabi.shouldCreateSourceAbiDump() {
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001463 exportedIncludeDirs := library.exportedIncludeDirsForAbiCheck(ctx)
Colin Cross91ae5ec2024-10-01 14:03:40 -07001464 headerAbiChecker := library.getHeaderAbiCheckerProperties(ctx.Module().(*Module))
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001465 currSdkVersion := currRefAbiDumpSdkVersion(ctx)
1466 currVendorVersion := ctx.Config().VendorApiLevel()
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001467
1468 // Generate source dumps.
1469 implDump := transformDumpToLinkedDump(ctx,
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001470 objs.sAbiDumpFiles, soFile, fileName,
1471 exportedIncludeDirs,
Hsin-Yi Chen843c0632023-09-25 15:26:30 +08001472 android.OptionalPathForModuleSrc(ctx, library.symbolFileForAbiCheck(ctx)),
1473 headerAbiChecker.Exclude_symbol_versions,
1474 headerAbiChecker.Exclude_symbol_tags,
Hsin-Yi Chenea748242025-01-15 15:44:48 +08001475 []string{} /* includeSymbolTags */, currSdkVersion, requiresGlobalIncludes(ctx))
Hsin-Yi Chen843c0632023-09-25 15:26:30 +08001476
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001477 var llndkDump, apexVariantDump android.Path
Colin Cross91ae5ec2024-10-01 14:03:40 -07001478 tags := classifySourceAbiDump(ctx.Module().(*Module))
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001479 optInTags := []lsdumpTag{}
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001480 for _, tag := range tags {
Hsin-Yi Chenf6fc5252024-04-15 15:43:21 +08001481 if tag == llndkLsdumpTag && currVendorVersion != "" {
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001482 if llndkDump == nil {
Hsin-Yi Chend6aaadc2024-11-13 16:58:46 +08001483 sdkVersion, err := android.GetSdkVersionForVendorApiLevel(currVendorVersion)
1484 if err != nil {
1485 ctx.ModuleErrorf("Cannot create %s llndk dump: %s", fileName, err)
1486 return
1487 }
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001488 // TODO(b/323447559): Evaluate if replacing sAbiDumpFiles with implDump is faster
1489 llndkDump = library.linkLlndkSAbiDumpFiles(ctx,
1490 deps, objs.sAbiDumpFiles, soFile, fileName,
1491 headerAbiChecker.Exclude_symbol_versions,
Hsin-Yi Chenf6fc5252024-04-15 15:43:21 +08001492 headerAbiChecker.Exclude_symbol_tags,
Hsin-Yi Chend6aaadc2024-11-13 16:58:46 +08001493 nativeClampedApiLevel(ctx, sdkVersion).String())
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001494 }
Colin Cross50bc3bc2024-10-25 09:53:41 -07001495 addLsdumpPath(ctx.Config(), string(tag)+":"+llndkDump.String())
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001496 } else if tag == apexLsdumpTag {
1497 if apexVariantDump == nil {
1498 apexVariantDump = library.linkApexSAbiDumpFiles(ctx,
1499 deps, objs.sAbiDumpFiles, soFile, fileName,
1500 headerAbiChecker.Exclude_symbol_versions,
1501 headerAbiChecker.Exclude_symbol_tags,
1502 currSdkVersion)
1503 }
Colin Cross50bc3bc2024-10-25 09:53:41 -07001504 addLsdumpPath(ctx.Config(), string(tag)+":"+apexVariantDump.String())
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001505 } else {
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001506 if tag.dirName() == "" {
1507 optInTags = append(optInTags, tag)
1508 }
Colin Cross50bc3bc2024-10-25 09:53:41 -07001509 addLsdumpPath(ctx.Config(), string(tag)+":"+implDump.String())
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001510 }
1511 }
1512
1513 // Diff source dumps and reference dumps.
1514 for _, tag := range tags {
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001515 dumpDirName := tag.dirName()
1516 if dumpDirName == "" {
1517 continue
1518 }
1519 dumpDir := filepath.Join("prebuilts", "abi-dumps", dumpDirName)
1520 isLlndk := (tag == llndkLsdumpTag)
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001521 isApex := (tag == apexLsdumpTag)
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001522 binderBitness := ctx.DeviceConfig().BinderBitness()
1523 nameExt := ""
1524 if isLlndk {
1525 nameExt = "llndk"
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001526 } else if isApex {
1527 nameExt = "apex"
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001528 }
1529 // Check against the previous version.
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001530 var prevVersion, currVersion string
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001531 sourceDump := implDump
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001532 // If this release config does not define VendorApiLevel, fall back to the old policy.
1533 if isLlndk && currVendorVersion != "" {
1534 prevVersion = ctx.Config().PrevVendorApiLevel()
1535 currVersion = currVendorVersion
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001536 // LLNDK dumps are generated by different rules after trunk stable.
1537 if android.IsTrunkStableVendorApiLevel(prevVersion) {
1538 sourceDump = llndkDump
1539 }
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001540 } else {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001541 prevVersionInt, currVersionInt := crossVersionAbiDiffSdkVersions(ctx, dumpDir)
1542 prevVersion = strconv.Itoa(prevVersionInt)
1543 currVersion = strconv.Itoa(currVersionInt)
1544 // APEX dumps are generated by different rules after trunk stable.
1545 if isApex && prevVersionInt > 34 {
1546 sourceDump = apexVariantDump
1547 }
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001548 }
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001549 prevDumpDir := filepath.Join(dumpDir, prevVersion, binderBitness)
1550 prevDumpFile := getRefAbiDumpFile(ctx, prevDumpDir, fileName)
1551 if prevDumpFile.Valid() {
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001552 library.crossVersionAbiDiff(ctx, sourceDump, prevDumpFile.Path(),
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001553 fileName, nameExt+prevVersion, isLlndk, currVersion, prevDumpDir)
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001554 }
1555 // Check against the current version.
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001556 sourceDump = implDump
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001557 if isLlndk && currVendorVersion != "" {
1558 currVersion = currVendorVersion
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001559 if android.IsTrunkStableVendorApiLevel(currVersion) {
1560 sourceDump = llndkDump
1561 }
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001562 } else {
1563 currVersion = currSdkVersion
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001564 if isApex && (!ctx.Config().PlatformSdkFinal() ||
1565 ctx.Config().PlatformSdkVersion().FinalInt() > 34) {
1566 sourceDump = apexVariantDump
1567 }
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001568 }
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001569 currDumpDir := filepath.Join(dumpDir, currVersion, binderBitness)
1570 currDumpFile := getRefAbiDumpFile(ctx, currDumpDir, fileName)
1571 if currDumpFile.Valid() {
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001572 library.sameVersionAbiDiff(ctx, sourceDump, currDumpFile.Path(),
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001573 fileName, nameExt, isLlndk, string(tag))
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001574 }
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -08001575 }
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001576
1577 // Assert that a module is tagged with at most one of platformLsdumpTag, productLsdumpTag, or vendorLsdumpTag.
1578 if len(headerAbiChecker.Ref_dump_dirs) > 0 && len(optInTags) != 1 {
1579 ctx.ModuleErrorf("Expect exactly one opt-in lsdump tag when ref_dump_dirs are specified: %s", optInTags)
1580 return
1581 }
Hsin-Yi Chen3d5c6792024-04-22 11:47:57 +08001582 // Ensure that a module tagged with only platformLsdumpTag has ref_dump_dirs.
1583 // Android.bp in vendor projects should be cleaned up before this is enforced for vendorLsdumpTag and productLsdumpTag.
1584 if len(headerAbiChecker.Ref_dump_dirs) == 0 && len(tags) == 1 && tags[0] == platformLsdumpTag {
1585 ctx.ModuleErrorf("header_abi_checker is explicitly enabled, but no ref_dump_dirs are specified.")
1586 }
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001587 // Check against the opt-in reference dumps.
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +08001588 for i, optInDumpDir := range headerAbiChecker.Ref_dump_dirs {
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001589 optInDumpDirPath := android.PathForModuleSrc(ctx, optInDumpDir)
1590 // Ref_dump_dirs are not versioned.
1591 // They do not contain subdir for binder bitness because 64-bit binder has been mandatory.
1592 optInDumpFile := getRefAbiDumpFile(ctx, optInDumpDirPath.String(), fileName)
1593 if !optInDumpFile.Valid() {
1594 continue
1595 }
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001596 library.optInAbiDiff(ctx,
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001597 implDump, optInDumpFile.Path(),
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001598 fileName, "opt"+strconv.Itoa(i), optInDumpDirPath.String(), string(optInTags[0]))
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001599 }
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -08001600 }
1601}
1602
Chris Parsons3c27ca32020-11-20 12:42:07 -05001603// link registers actions to link this library, and sets various fields
1604// on this library to reflect information that should be exported up the build
1605// tree (for example, exported flags and include paths).
Colin Crossb916a382016-07-29 17:28:03 -07001606func (library *libraryDecorator) link(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001607 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Cross4d9c2d12016-07-29 12:48:20 -07001608
Colin Cross127bb8b2020-12-16 16:46:01 -08001609 if ctx.IsLlndk() {
Colin Cross0fb7fcd2021-03-02 11:00:07 -08001610 // override the module's export_include_dirs with llndk.override_export_include_dirs
1611 // if it is set.
1612 if override := library.Properties.Llndk.Override_export_include_dirs; override != nil {
Aleks Todorovc9becde2024-06-10 12:51:53 +01001613 library.flagExporter.Properties.Export_include_dirs = proptools.NewConfigurable[[]string](
1614 nil,
1615 []proptools.ConfigurableCase[[]string]{
1616 proptools.NewConfigurableCase[[]string](nil, &override),
1617 },
1618 )
Colin Cross0fb7fcd2021-03-02 11:00:07 -08001619 }
1620
Colin Cross127bb8b2020-12-16 16:46:01 -08001621 if Bool(library.Properties.Llndk.Export_headers_as_system) {
1622 library.flagExporter.Properties.Export_system_include_dirs = append(
1623 library.flagExporter.Properties.Export_system_include_dirs,
Aleks Todorovc9becde2024-06-10 12:51:53 +01001624 library.flagExporter.Properties.Export_include_dirs.GetOrDefault(ctx, nil)...)
1625 library.flagExporter.Properties.Export_include_dirs = proptools.NewConfigurable[[]string](nil, nil)
Colin Cross127bb8b2020-12-16 16:46:01 -08001626 }
1627 }
1628
Colin Cross5271fea2021-04-27 13:06:04 -07001629 if ctx.IsVendorPublicLibrary() {
1630 // override the module's export_include_dirs with vendor_public_library.override_export_include_dirs
1631 // if it is set.
1632 if override := library.Properties.Vendor_public_library.Override_export_include_dirs; override != nil {
Aleks Todorovc9becde2024-06-10 12:51:53 +01001633 library.flagExporter.Properties.Export_include_dirs = proptools.NewConfigurable[[]string](
1634 nil,
1635 []proptools.ConfigurableCase[[]string]{
1636 proptools.NewConfigurableCase[[]string](nil, &override),
1637 },
1638 )
Colin Cross5271fea2021-04-27 13:06:04 -07001639 }
1640 }
1641
Chris Parsons3c27ca32020-11-20 12:42:07 -05001642 // Linking this library consists of linking `deps.Objs` (.o files in dependencies
1643 // of this library), together with `objs` (.o files created by compiling this
1644 // library).
Colin Crossad59e752017-11-16 14:29:11 -08001645 objs = deps.Objs.Copy().Append(objs)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001646 var out android.Path
Colin Crossa48ab5b2017-02-14 15:28:44 -08001647 if library.static() || library.header() {
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001648 out = library.linkStatic(ctx, flags, deps, objs)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001649 } else {
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001650 out = library.linkShared(ctx, flags, deps, objs)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001651 }
1652
Chris Parsons3c27ca32020-11-20 12:42:07 -05001653 // Export include paths and flags to be propagated up the tree.
Inseob Kim69378442019-06-03 19:10:47 +09001654 library.exportIncludes(ctx)
1655 library.reexportDirs(deps.ReexportedDirs...)
1656 library.reexportSystemDirs(deps.ReexportedSystemDirs...)
1657 library.reexportFlags(deps.ReexportedFlags...)
1658 library.reexportDeps(deps.ReexportedDeps...)
Inseob Kimd110f872019-12-06 13:15:38 +09001659 library.addExportedGeneratedHeaders(deps.ReexportedGeneratedHeaders...)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001660
Ivan Lozano0a468a42024-05-13 21:03:34 -04001661 if library.static() && len(deps.ReexportedRustRlibDeps) > 0 {
1662 library.reexportRustStaticDeps(deps.ReexportedRustRlibDeps...)
1663 }
1664
Chris Parsons3c27ca32020-11-20 12:42:07 -05001665 // Optionally export aidl headers.
Nan Zhang0007d812017-11-07 10:57:05 -08001666 if Bool(library.Properties.Aidl.Export_aidl_headers) {
Cole Faust96a692b2024-08-08 14:47:51 -07001667 if library.baseCompiler.hasAidl(ctx, deps) {
1668 if library.baseCompiler.hasSrcExt(ctx, ".aidl") {
Vinh Tran09581952023-05-16 16:03:20 -04001669 dir := android.PathForModuleGen(ctx, "aidl")
1670 library.reexportDirs(dir)
1671 }
1672 if len(deps.AidlLibraryInfos) > 0 {
1673 dir := android.PathForModuleGen(ctx, "aidl_library")
1674 library.reexportDirs(dir)
1675 }
Inseob Kimd110f872019-12-06 13:15:38 +09001676
Paul Duffin33056e82021-02-19 13:49:08 +00001677 library.reexportDeps(library.baseCompiler.aidlOrderOnlyDeps...)
1678 library.addExportedGeneratedHeaders(library.baseCompiler.aidlHeaders...)
Dan Willemsene1240db2016-11-03 14:28:51 -07001679 }
1680 }
1681
Chris Parsons3c27ca32020-11-20 12:42:07 -05001682 // Optionally export proto headers.
Nan Zhang0007d812017-11-07 10:57:05 -08001683 if Bool(library.Properties.Proto.Export_proto_headers) {
Cole Faust96a692b2024-08-08 14:47:51 -07001684 if library.baseCompiler.hasSrcExt(ctx, ".proto") {
Jiyong Park74955042019-10-22 20:19:51 +09001685 var includes android.Paths
Colin Cross19878da2019-03-28 14:45:07 -07001686 if flags.proto.CanonicalPathFromRoot {
Jiyong Park74955042019-10-22 20:19:51 +09001687 includes = append(includes, flags.proto.SubDir)
Colin Cross10d22312017-05-03 11:01:58 -07001688 }
Jiyong Park74955042019-10-22 20:19:51 +09001689 includes = append(includes, flags.proto.Dir)
Inseob Kim69378442019-06-03 19:10:47 +09001690 library.reexportDirs(includes...)
Inseob Kimd110f872019-12-06 13:15:38 +09001691
Paul Duffin33056e82021-02-19 13:49:08 +00001692 library.reexportDeps(library.baseCompiler.protoOrderOnlyDeps...)
1693 library.addExportedGeneratedHeaders(library.baseCompiler.protoHeaders...)
Colin Cross0c461f12016-10-20 16:11:43 -07001694 }
1695 }
1696
Inseob Kim07def122020-11-23 14:43:02 +09001697 // If the library is sysprop_library, expose either public or internal header selectively.
Cole Faust96a692b2024-08-08 14:47:51 -07001698 if library.baseCompiler.hasSrcExt(ctx, ".sysprop") {
Jiyong Park74955042019-10-22 20:19:51 +09001699 dir := android.PathForModuleGen(ctx, "sysprop", "include")
Inseob Kimc0907f12019-02-08 21:00:45 +09001700 if library.Properties.Sysprop.Platform != nil {
Inseob Kimc0907f12019-02-08 21:00:45 +09001701 isOwnerPlatform := Bool(library.Properties.Sysprop.Platform)
1702
Inseob Kim07def122020-11-23 14:43:02 +09001703 // If the owner is different from the user, expose public header. That is,
1704 // 1) if the user is product (as owner can only be platform / vendor)
Inseob Kim06410042021-01-18 15:23:28 +09001705 // 2) if the owner is platform and the client is vendor
1706 // We don't care Platform -> Vendor dependency as it's already forbidden.
1707 if ctx.Device() && (ctx.ProductSpecific() || (isOwnerPlatform && ctx.inVendor())) {
Jiyong Park74955042019-10-22 20:19:51 +09001708 dir = android.PathForModuleGen(ctx, "sysprop/public", "include")
Inseob Kimc0907f12019-02-08 21:00:45 +09001709 }
1710 }
1711
Paul Duffin37e0de52021-02-19 17:05:39 +00001712 // Make sure to only export headers which are within the include directory.
1713 _, headers := android.FilterPathListPredicate(library.baseCompiler.syspropHeaders, func(path android.Path) bool {
1714 _, isRel := android.MaybeRel(ctx, dir.String(), path.String())
1715 return isRel
1716 })
1717
Chris Parsons3c27ca32020-11-20 12:42:07 -05001718 // Add sysprop-related directories to the exported directories of this library.
Inseob Kim69378442019-06-03 19:10:47 +09001719 library.reexportDirs(dir)
Paul Duffin33056e82021-02-19 13:49:08 +00001720 library.reexportDeps(library.baseCompiler.syspropOrderOnlyDeps...)
Paul Duffin37e0de52021-02-19 17:05:39 +00001721 library.addExportedGeneratedHeaders(headers...)
Inseob Kim21f26902018-09-06 00:55:20 +09001722 }
1723
Chris Parsons3c27ca32020-11-20 12:42:07 -05001724 // Add stub-related flags if this library is a stub library.
Jiyong Park892a98f2020-12-14 09:20:00 +09001725 library.exportVersioningMacroIfNeeded(ctx)
Jiyong Parkda732bd2018-11-02 18:23:15 +09001726
Chris Parsons3c27ca32020-11-20 12:42:07 -05001727 // Propagate a Provider containing information about exported flags, deps, and include paths.
Colin Cross0de8a1e2020-09-18 14:15:30 -07001728 library.flagExporter.setProvider(ctx)
1729
Colin Cross4d9c2d12016-07-29 12:48:20 -07001730 return out
1731}
1732
Jiyong Park892a98f2020-12-14 09:20:00 +09001733func (library *libraryDecorator) exportVersioningMacroIfNeeded(ctx android.BaseModuleContext) {
Colin Cross127bb8b2020-12-16 16:46:01 -08001734 if library.buildStubs() && library.stubsVersion() != "" && !library.skipAPIDefine {
Jiyong Park892a98f2020-12-14 09:20:00 +09001735 name := versioningMacroName(ctx.Module().(*Module).ImplementationModuleName(ctx))
Jooyung Han11b0fbd2021-02-05 02:28:22 +09001736 apiLevel, err := android.ApiLevelFromUser(ctx, library.stubsVersion())
1737 if err != nil {
1738 ctx.ModuleErrorf("Can't export version macro: %s", err.Error())
1739 }
1740 library.reexportFlags("-D" + name + "=" + strconv.Itoa(apiLevel.FinalOrPreviewInt()))
Jiyong Park892a98f2020-12-14 09:20:00 +09001741 }
1742}
1743
Chris Parsons3c27ca32020-11-20 12:42:07 -05001744// buildStatic returns true if this library should be built as a static library.
Colin Crossb916a382016-07-29 17:28:03 -07001745func (library *libraryDecorator) buildStatic() bool {
Colin Crosse1bb5d02019-09-24 14:55:04 -07001746 return library.MutatedProperties.BuildStatic &&
1747 BoolDefault(library.StaticProperties.Static.Enabled, true)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001748}
1749
Chris Parsons3c27ca32020-11-20 12:42:07 -05001750// buildShared returns true if this library should be built as a shared library.
Colin Crossb916a382016-07-29 17:28:03 -07001751func (library *libraryDecorator) buildShared() bool {
Colin Crosse1bb5d02019-09-24 14:55:04 -07001752 return library.MutatedProperties.BuildShared &&
1753 BoolDefault(library.SharedProperties.Shared.Enabled, true)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001754}
1755
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001756func (library *libraryDecorator) objs() Objects {
1757 return library.objects
Colin Cross4d9c2d12016-07-29 12:48:20 -07001758}
1759
Colin Cross0de8a1e2020-09-18 14:15:30 -07001760func (library *libraryDecorator) reuseObjs() Objects {
1761 return library.reuseObjects
Colin Cross4d9c2d12016-07-29 12:48:20 -07001762}
1763
Colin Cross26c34ed2016-09-30 17:10:16 -07001764func (library *libraryDecorator) toc() android.OptionalPath {
1765 return library.tocFile
1766}
1767
Jiyong Parkf1194352019-02-25 11:05:47 +09001768func (library *libraryDecorator) installSymlinkToRuntimeApex(ctx ModuleContext, file android.Path) {
1769 dir := library.baseInstaller.installDir(ctx)
1770 dirOnDevice := android.InstallPathToOnDevicePath(ctx, dir)
Florian Mayer95cd6db2023-03-23 17:48:07 -07001771 // libc_hwasan has relative_install_dir set, which would mess up the dir.Base() logic.
1772 // hardcode here because it's the only target, if we have other targets that use this
1773 // we can generalise this.
1774 var target string
1775 if ctx.baseModuleName() == "libc_hwasan" {
1776 target = "/" + filepath.Join("apex", "com.android.runtime", "lib64", "bionic", "hwasan", file.Base())
1777 } else {
1778 base := dir.Base()
1779 target = "/" + filepath.Join("apex", "com.android.runtime", base, "bionic", file.Base())
1780 }
Jiyong Parkf1194352019-02-25 11:05:47 +09001781 ctx.InstallAbsoluteSymlink(dir, file.Base(), target)
Jaewoong Jung18aefc12020-12-21 09:11:10 -08001782 library.postInstallCmds = append(library.postInstallCmds, makeSymlinkCmd(dirOnDevice, file.Base(), target))
Jiyong Parkf1194352019-02-25 11:05:47 +09001783}
1784
Colin Crossb916a382016-07-29 17:28:03 -07001785func (library *libraryDecorator) install(ctx ModuleContext, file android.Path) {
Colin Crossc43ae772017-04-14 15:42:53 -07001786 if library.shared() {
Colin Cross386afeb2024-11-13 15:20:13 -08001787 translatedArch := ctx.Target().NativeBridge == android.NativeBridgeEnabled
1788 if library.hasStubsVariants() && !ctx.Host() && !ctx.isSdkVariant() &&
1789 InstallToBootstrap(ctx.baseModuleName(), ctx.Config()) && !library.buildStubs() &&
1790 !translatedArch && !ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() {
Jiyong Parkf1194352019-02-25 11:05:47 +09001791 // Bionic libraries (e.g. libc.so) is installed to the bootstrap subdirectory.
1792 // The original path becomes a symlink to the corresponding file in the
1793 // runtime APEX.
Colin Cross386afeb2024-11-13 15:20:13 -08001794 if ctx.Device() {
1795 library.installSymlinkToRuntimeApex(ctx, file)
Jiyong Park429660f2019-01-16 22:31:11 +09001796 }
Colin Cross386afeb2024-11-13 15:20:13 -08001797 library.baseInstaller.subDir = "bootstrap"
Justin Yun8effde42017-06-23 19:24:43 +09001798 }
Przemyslaw Szczepaniake6fc5072019-07-12 14:06:23 +01001799
Colin Cross4d9c2d12016-07-29 12:48:20 -07001800 library.baseInstaller.install(ctx, file)
1801 }
Dan Albertf563d252017-10-13 00:29:00 -07001802
Dan Albert281f22b2017-12-13 15:03:47 -08001803 if Bool(library.Properties.Static_ndk_lib) && library.static() &&
Kiyoung Kimaa394802024-01-08 12:55:45 +09001804 !ctx.InVendorOrProduct() && !ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() && ctx.Device() &&
Jiyong Park7ed9de32018-10-15 22:25:07 +09001805 library.baseLinker.sanitize.isUnsanitizedVariant() &&
Colin Cross95b07f22020-12-16 11:06:50 -08001806 ctx.isForPlatform() && !ctx.isPreventInstall() {
Dan Albert4048bb02023-04-03 20:19:07 +00001807 installPath := getUnversionedLibraryInstallPath(ctx).Join(ctx, file.Base())
Dan Albertf563d252017-10-13 00:29:00 -07001808
1809 ctx.ModuleBuild(pctx, android.ModuleBuildParams{
1810 Rule: android.Cp,
1811 Description: "install " + installPath.Base(),
1812 Output: installPath,
1813 Input: file,
1814 })
1815
Colin Cross0875c522017-11-28 17:34:01 -08001816 library.ndkSysrootPath = installPath
Dan Albertf563d252017-10-13 00:29:00 -07001817 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001818}
1819
Paul Duffin0cb37b92020-03-04 14:52:46 +00001820func (library *libraryDecorator) everInstallable() bool {
1821 // Only shared and static libraries are installed. Header libraries (which are
1822 // neither static or shared) are not installed.
1823 return library.shared() || library.static()
1824}
1825
Colin Cross6a730042024-12-05 13:53:43 -08001826// static returns true if this library is for a "static" variant.
Colin Crossb916a382016-07-29 17:28:03 -07001827func (library *libraryDecorator) static() bool {
Colin Crossa48ab5b2017-02-14 15:28:44 -08001828 return library.MutatedProperties.VariantIsStatic
Colin Cross4d9c2d12016-07-29 12:48:20 -07001829}
1830
Colin Cross6a730042024-12-05 13:53:43 -08001831// staticLibrary returns true if this library is for a "static"" variant.
1832func (library *libraryDecorator) staticLibrary() bool {
1833 return library.static()
1834}
1835
1836// shared returns true if this library is for a "shared" variant.
Colin Crossa48ab5b2017-02-14 15:28:44 -08001837func (library *libraryDecorator) shared() bool {
1838 return library.MutatedProperties.VariantIsShared
1839}
1840
Chris Parsons3c27ca32020-11-20 12:42:07 -05001841// header returns true if this library is for a header-only variant.
Colin Crossa48ab5b2017-02-14 15:28:44 -08001842func (library *libraryDecorator) header() bool {
Chris Parsons3c27ca32020-11-20 12:42:07 -05001843 // Neither "static" nor "shared" implies this library is header-only.
Colin Crossa48ab5b2017-02-14 15:28:44 -08001844 return !library.static() && !library.shared()
1845}
1846
Chris Parsons3c27ca32020-11-20 12:42:07 -05001847// setStatic marks the library variant as "static".
Colin Crossa48ab5b2017-02-14 15:28:44 -08001848func (library *libraryDecorator) setStatic() {
1849 library.MutatedProperties.VariantIsStatic = true
1850 library.MutatedProperties.VariantIsShared = false
1851}
1852
Chris Parsons3c27ca32020-11-20 12:42:07 -05001853// setShared marks the library variant as "shared".
Colin Crossa48ab5b2017-02-14 15:28:44 -08001854func (library *libraryDecorator) setShared() {
1855 library.MutatedProperties.VariantIsStatic = false
1856 library.MutatedProperties.VariantIsShared = true
Colin Crossb916a382016-07-29 17:28:03 -07001857}
1858
Chris Parsons3c27ca32020-11-20 12:42:07 -05001859// BuildOnlyStatic disables building this library as a shared library.
Colin Crossab3b7322016-12-09 14:46:15 -08001860func (library *libraryDecorator) BuildOnlyStatic() {
Colin Crossa48ab5b2017-02-14 15:28:44 -08001861 library.MutatedProperties.BuildShared = false
Colin Crossab3b7322016-12-09 14:46:15 -08001862}
1863
Chris Parsons3c27ca32020-11-20 12:42:07 -05001864// BuildOnlyShared disables building this library as a static library.
Colin Crossab3b7322016-12-09 14:46:15 -08001865func (library *libraryDecorator) BuildOnlyShared() {
Colin Crossa48ab5b2017-02-14 15:28:44 -08001866 library.MutatedProperties.BuildStatic = false
Colin Crossab3b7322016-12-09 14:46:15 -08001867}
1868
Chris Parsons3c27ca32020-11-20 12:42:07 -05001869// HeaderOnly disables building this library as a shared or static library;
1870// the library only exists to propagate header file dependencies up the build graph.
Colin Cross5950f382016-12-13 12:50:57 -08001871func (library *libraryDecorator) HeaderOnly() {
Colin Crossa48ab5b2017-02-14 15:28:44 -08001872 library.MutatedProperties.BuildShared = false
1873 library.MutatedProperties.BuildStatic = false
Colin Cross5950f382016-12-13 12:50:57 -08001874}
1875
Colin Cross127bb8b2020-12-16 16:46:01 -08001876// hasLLNDKStubs returns true if this cc_library module has a variant that will build LLNDK stubs.
1877func (library *libraryDecorator) hasLLNDKStubs() bool {
Colin Cross203b4212021-04-26 17:19:41 -07001878 return String(library.Properties.Llndk.Symbol_file) != ""
Colin Cross0fb7fcd2021-03-02 11:00:07 -08001879}
1880
Colin Cross203b4212021-04-26 17:19:41 -07001881// hasLLNDKStubs returns true if this cc_library module has a variant that will build LLNDK stubs.
Colin Cross1f3f1302021-04-26 18:37:44 -07001882func (library *libraryDecorator) hasLLNDKHeaders() bool {
1883 return Bool(library.Properties.Llndk.Llndk_headers)
1884}
1885
Colin Cross0510f542024-11-13 16:07:08 -08001886// isLLNDKMovedToApex returns true if this cc_library module sets the llndk.moved_to_apex property.
1887func (library *libraryDecorator) isLLNDKMovedToApex() bool {
1888 return Bool(library.Properties.Llndk.Moved_to_apex)
1889}
1890
Colin Cross5271fea2021-04-27 13:06:04 -07001891// hasVendorPublicLibrary returns true if this cc_library module has a variant that will build
1892// vendor public library stubs.
1893func (library *libraryDecorator) hasVendorPublicLibrary() bool {
1894 return String(library.Properties.Vendor_public_library.Symbol_file) != ""
1895}
1896
Colin Cross0477b422020-10-13 18:43:54 -07001897func (library *libraryDecorator) implementationModuleName(name string) string {
1898 return name
1899}
1900
Jiyong Park7ed9de32018-10-15 22:25:07 +09001901func (library *libraryDecorator) buildStubs() bool {
1902 return library.MutatedProperties.BuildStubs
1903}
1904
Jiyong Parka034b832019-08-27 14:02:19 +09001905func (library *libraryDecorator) symbolFileForAbiCheck(ctx ModuleContext) *string {
Colin Cross91ae5ec2024-10-01 14:03:40 -07001906 if props := library.getHeaderAbiCheckerProperties(ctx.Module().(*Module)); props.Symbol_file != nil {
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +08001907 return props.Symbol_file
Jiyong Parka034b832019-08-27 14:02:19 +09001908 }
Colin Cross31076b32020-10-23 17:22:06 -07001909 if library.hasStubsVariants() && library.Properties.Stubs.Symbol_file != nil {
Jiyong Parka034b832019-08-27 14:02:19 +09001910 return library.Properties.Stubs.Symbol_file
1911 }
Hsin-Yi Chenf1f276c2024-03-11 20:02:34 +08001912 // TODO(b/309880485): Distinguish platform, NDK, LLNDK, and APEX version scripts.
1913 if library.baseLinker.Properties.Version_script != nil {
1914 return library.baseLinker.Properties.Version_script
1915 }
Jiyong Parka034b832019-08-27 14:02:19 +09001916 return nil
1917}
1918
Colin Crossc88c2722020-09-28 17:32:47 -07001919func (library *libraryDecorator) hasStubsVariants() bool {
Jiyong Parkd4a3a132021-03-17 20:21:35 +09001920 // Just having stubs.symbol_file is enough to create a stub variant. In that case
1921 // the stub for the future API level is created.
1922 return library.Properties.Stubs.Symbol_file != nil ||
1923 len(library.Properties.Stubs.Versions) > 0
Colin Crossc88c2722020-09-28 17:32:47 -07001924}
1925
Alan Stokes73feba32022-11-14 12:21:24 +00001926func (library *libraryDecorator) isStubsImplementationRequired() bool {
1927 return BoolDefault(library.Properties.Stubs.Implementation_installable, true)
1928}
1929
Colin Crossadd04a82024-05-22 09:57:59 -07001930func (library *libraryDecorator) stubsVersions(ctx android.BaseModuleContext) []string {
Jiyong Parkd4a3a132021-03-17 20:21:35 +09001931 if !library.hasStubsVariants() {
1932 return nil
1933 }
1934
Kiyoung Kimaa394802024-01-08 12:55:45 +09001935 if library.hasLLNDKStubs() && ctx.Module().(*Module).InVendorOrProduct() {
Jooyung Han5e8994e2024-03-12 14:12:12 +09001936 // LLNDK libraries only need a single stubs variant (""), which is
1937 // added automatically in createVersionVariations().
1938 return nil
Colin Crossc1b36442021-05-06 13:42:48 -07001939 }
1940
Jiyong Parkd4a3a132021-03-17 20:21:35 +09001941 // Future API level is implicitly added if there isn't
Jooyung Han5e8994e2024-03-12 14:12:12 +09001942 versions := addCurrentVersionIfNotPresent(library.Properties.Stubs.Versions)
1943 normalizeVersions(ctx, versions)
1944 return versions
Sam Delmerico75dbca22023-04-20 13:13:25 +00001945}
1946
1947func addCurrentVersionIfNotPresent(vers []string) []string {
Jiyong Parkd4a3a132021-03-17 20:21:35 +09001948 if inList(android.FutureApiLevel.String(), vers) {
1949 return vers
1950 }
1951 // In some cases, people use the raw value "10000" in the versions property.
1952 // We shouldn't add the future API level in that case, otherwise there will
1953 // be two identical versions.
1954 if inList(strconv.Itoa(android.FutureApiLevel.FinalOrFutureInt()), vers) {
1955 return vers
1956 }
1957 return append(vers, android.FutureApiLevel.String())
Colin Crossc88c2722020-09-28 17:32:47 -07001958}
1959
1960func (library *libraryDecorator) setStubsVersion(version string) {
1961 library.MutatedProperties.StubsVersion = version
1962}
1963
Jiyong Park7ed9de32018-10-15 22:25:07 +09001964func (library *libraryDecorator) stubsVersion() string {
1965 return library.MutatedProperties.StubsVersion
1966}
1967
Jiyong Parkd4a3a132021-03-17 20:21:35 +09001968func (library *libraryDecorator) setBuildStubs(isLatest bool) {
Colin Crossc88c2722020-09-28 17:32:47 -07001969 library.MutatedProperties.BuildStubs = true
Jiyong Parkd4a3a132021-03-17 20:21:35 +09001970 library.MutatedProperties.IsLatestVersion = isLatest
Colin Crossc88c2722020-09-28 17:32:47 -07001971}
1972
1973func (library *libraryDecorator) setAllStubsVersions(versions []string) {
1974 library.MutatedProperties.AllStubsVersions = versions
1975}
1976
1977func (library *libraryDecorator) allStubsVersions() []string {
1978 return library.MutatedProperties.AllStubsVersions
1979}
1980
Jooyung Hanad4c1872020-02-27 17:56:44 +09001981func (library *libraryDecorator) isLatestStubVersion() bool {
Jiyong Parkd4a3a132021-03-17 20:21:35 +09001982 return library.MutatedProperties.IsLatestVersion
Jooyung Hanad4c1872020-02-27 17:56:44 +09001983}
1984
Yu Liub73c3a62024-12-10 00:58:06 +00001985func (library *libraryDecorator) apexAvailable() []string {
Jiyong Parka90ca002019-10-07 15:47:24 +09001986 var list []string
1987 if library.static() {
1988 list = library.StaticProperties.Static.Apex_available
1989 } else if library.shared() {
1990 list = library.SharedProperties.Shared.Apex_available
1991 }
Yu Liub73c3a62024-12-10 00:58:06 +00001992
1993 return list
Jiyong Parka90ca002019-10-07 15:47:24 +09001994}
1995
Colin Cross1bc94122021-10-28 13:25:54 -07001996func (library *libraryDecorator) installable() *bool {
1997 if library.static() {
1998 return library.StaticProperties.Static.Installable
1999 } else if library.shared() {
2000 return library.SharedProperties.Shared.Installable
2001 }
2002 return nil
2003}
2004
Jingwen Chen8ac7d7d2023-03-20 11:05:16 +00002005func (library *libraryDecorator) makeUninstallable(mod *Module) {
2006 if library.static() && library.buildStatic() && !library.buildStubs() {
2007 // If we're asked to make a static library uninstallable we don't do
2008 // anything since AndroidMkEntries always sets LOCAL_UNINSTALLABLE_MODULE
2009 // for these entries. This is done to still get the make targets for NOTICE
2010 // files from notice_files.mk, which other libraries might depend on.
2011 return
2012 }
2013 mod.ModuleBase.MakeUninstallable()
2014}
2015
Jihoon Kangf78a8902022-09-01 22:47:07 +00002016func (library *libraryDecorator) getPartition() string {
2017 return library.path.Partition()
2018}
2019
Colin Crossceaa5322021-09-28 16:37:50 -07002020func (library *libraryDecorator) getAPIListCoverageXMLPath() android.ModuleOutPath {
2021 return library.apiListCoverageXmlPath
2022}
2023
Inseob Kima1888ce2022-10-04 14:42:02 +09002024func (library *libraryDecorator) overriddenModules() []string {
2025 return library.Properties.Overrides
2026}
2027
2028var _ overridable = (*libraryDecorator)(nil)
2029
Colin Cross571cccf2019-02-04 11:22:08 -08002030var versioningMacroNamesListKey = android.NewOnceKey("versioningMacroNamesList")
2031
Chris Parsons3c27ca32020-11-20 12:42:07 -05002032// versioningMacroNamesList returns a singleton map, where keys are "version macro names",
2033// and values are the module name responsible for registering the version macro name.
2034//
2035// Version macros are used when building against stubs, to provide version information about
2036// the stub. Only stub libraries should have an entry in this list.
2037//
2038// For example, when building against libFoo#ver, __LIBFOO_API__ macro is set to ver so
2039// that headers from libFoo can be conditionally compiled (this may hide APIs
2040// that are not available for the version).
2041//
2042// This map is used to ensure that there aren't conflicts between these version macro names.
Jiyong Parkda732bd2018-11-02 18:23:15 +09002043func versioningMacroNamesList(config android.Config) *map[string]string {
Colin Cross571cccf2019-02-04 11:22:08 -08002044 return config.Once(versioningMacroNamesListKey, func() interface{} {
Jiyong Parkda732bd2018-11-02 18:23:15 +09002045 m := make(map[string]string)
2046 return &m
2047 }).(*map[string]string)
2048}
2049
2050// alphanumeric and _ characters are preserved.
2051// other characters are all converted to _
2052var charsNotForMacro = regexp.MustCompile("[^a-zA-Z0-9_]+")
2053
Chris Parsons3c27ca32020-11-20 12:42:07 -05002054// versioningMacroName returns the canonical version macro name for the given module.
Jiyong Parkda732bd2018-11-02 18:23:15 +09002055func versioningMacroName(moduleName string) string {
2056 macroName := charsNotForMacro.ReplaceAllString(moduleName, "_")
Jooyung Hanb04a4992020-03-13 18:57:35 +09002057 macroName = strings.ToUpper(macroName)
Jiyong Parkda732bd2018-11-02 18:23:15 +09002058 return "__" + macroName + "_API__"
2059}
2060
Chris Parsons3c27ca32020-11-20 12:42:07 -05002061// NewLibrary builds and returns a new Module corresponding to a C++ library.
2062// Individual module implementations which comprise a C++ library (or something like
2063// a C++ library) should call this function, set some fields on the result, and
2064// then call the Init function.
Colin Crossab3b7322016-12-09 14:46:15 -08002065func NewLibrary(hod android.HostOrDeviceSupported) (*Module, *libraryDecorator) {
Colin Cross4d9c2d12016-07-29 12:48:20 -07002066 module := newModule(hod, android.MultilibBoth)
2067
Colin Crossb916a382016-07-29 17:28:03 -07002068 library := &libraryDecorator{
Colin Crossa48ab5b2017-02-14 15:28:44 -08002069 MutatedProperties: LibraryMutatedProperties{
Colin Crossab3b7322016-12-09 14:46:15 -08002070 BuildShared: true,
2071 BuildStatic: true,
Colin Cross4d9c2d12016-07-29 12:48:20 -07002072 },
Colin Crossb916a382016-07-29 17:28:03 -07002073 baseCompiler: NewBaseCompiler(),
Dan Albert61f32122018-07-26 14:00:24 -07002074 baseLinker: NewBaseLinker(module.sanitize),
Colin Crossb916a382016-07-29 17:28:03 -07002075 baseInstaller: NewBaseInstaller("lib", "lib64", InstallInSystem),
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -08002076 sabi: module.sabi,
Colin Cross4d9c2d12016-07-29 12:48:20 -07002077 }
2078
Colin Crossb916a382016-07-29 17:28:03 -07002079 module.compiler = library
2080 module.linker = library
2081 module.installer = library
Colin Cross31076b32020-10-23 17:22:06 -07002082 module.library = library
Colin Crossb916a382016-07-29 17:28:03 -07002083
2084 return module, library
2085}
2086
Colin Cross10d22312017-05-03 11:01:58 -07002087// connects a shared library to a static library in order to reuse its .o files to avoid
2088// compiling source files twice.
Colin Cross767819f2024-05-22 14:22:34 -07002089func reuseStaticLibrary(ctx android.BottomUpMutatorContext, shared *Module) {
2090 if sharedCompiler, ok := shared.compiler.(*libraryDecorator); ok {
Dan Willemsen3a26eef2018-12-03 15:25:46 -08002091
2092 // Check libraries in addition to cflags, since libraries may be exporting different
2093 // include directories.
Colin Cross767819f2024-05-22 14:22:34 -07002094 if len(sharedCompiler.StaticProperties.Static.Cflags.GetOrDefault(ctx, nil)) == 0 &&
2095 len(sharedCompiler.SharedProperties.Shared.Cflags.GetOrDefault(ctx, nil)) == 0 &&
Cole Faustf0006e72024-08-19 14:39:19 -07002096 len(sharedCompiler.StaticProperties.Static.Whole_static_libs.GetOrDefault(ctx, nil)) == 0 &&
2097 len(sharedCompiler.SharedProperties.Shared.Whole_static_libs.GetOrDefault(ctx, nil)) == 0 &&
2098 len(sharedCompiler.StaticProperties.Static.Static_libs.GetOrDefault(ctx, nil)) == 0 &&
2099 len(sharedCompiler.SharedProperties.Shared.Static_libs.GetOrDefault(ctx, nil)) == 0 &&
2100 len(sharedCompiler.StaticProperties.Static.Shared_libs.GetOrDefault(ctx, nil)) == 0 &&
2101 len(sharedCompiler.SharedProperties.Shared.Shared_libs.GetOrDefault(ctx, nil)) == 0 &&
Martin Stjernholm10566a02020-03-24 01:19:52 +00002102 // Compare System_shared_libs properties with nil because empty lists are
2103 // semantically significant for them.
Colin Cross767819f2024-05-22 14:22:34 -07002104 sharedCompiler.StaticProperties.Static.System_shared_libs == nil &&
Colin Crosse1bb5d02019-09-24 14:55:04 -07002105 sharedCompiler.SharedProperties.Shared.System_shared_libs == nil {
Colin Cross10d22312017-05-03 11:01:58 -07002106
Colin Cross767819f2024-05-22 14:22:34 -07002107 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, reuseObjTag, ctx.ModuleName())
Colin Cross10d22312017-05-03 11:01:58 -07002108 }
Colin Cross0de8a1e2020-09-18 14:15:30 -07002109
2110 // This dep is just to reference static variant from shared variant
Colin Cross767819f2024-05-22 14:22:34 -07002111 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticVariantTag, ctx.ModuleName())
Colin Cross10d22312017-05-03 11:01:58 -07002112 }
2113}
2114
Colin Cross767819f2024-05-22 14:22:34 -07002115// linkageTransitionMutator adds "static" or "shared" variants for modules depending
Chris Parsons3c27ca32020-11-20 12:42:07 -05002116// on whether the module can be built as a static library or a shared library.
Colin Cross767819f2024-05-22 14:22:34 -07002117type linkageTransitionMutator struct{}
2118
2119func (linkageTransitionMutator) Split(ctx android.BaseModuleContext) []string {
Jaewoong Jung18aefc12020-12-21 09:11:10 -08002120 ccPrebuilt := false
Colin Cross767819f2024-05-22 14:22:34 -07002121 if m, ok := ctx.Module().(*Module); ok && m.linker != nil {
Jaewoong Jung18aefc12020-12-21 09:11:10 -08002122 _, ccPrebuilt = m.linker.(prebuiltLibraryInterface)
Ivan Lozano52767be2019-10-18 14:49:46 -07002123 }
Jaewoong Jung18aefc12020-12-21 09:11:10 -08002124 if ccPrebuilt {
Colin Cross767819f2024-05-22 14:22:34 -07002125 library := ctx.Module().(*Module).linker.(prebuiltLibraryInterface)
Colin Cross33b2fb72019-05-14 14:07:01 -07002126
Paul Duffinf5ea9e12020-02-21 10:57:00 +00002127 // Differentiate between header only and building an actual static/shared library
Colin Cross127bb8b2020-12-16 16:46:01 -08002128 buildStatic := library.buildStatic()
2129 buildShared := library.buildShared()
2130 if buildStatic || buildShared {
Paul Duffinf5ea9e12020-02-21 10:57:00 +00002131 // Always create both the static and shared variants for prebuilt libraries, and then disable the one
2132 // that is not being used. This allows them to share the name of a cc_library module, which requires that
2133 // all the variants of the cc_library also exist on the prebuilt.
Colin Cross767819f2024-05-22 14:22:34 -07002134 return []string{"static", "shared"}
Paul Duffinf5ea9e12020-02-21 10:57:00 +00002135 } else {
2136 // Header only
Colin Crossb916a382016-07-29 17:28:03 -07002137 }
Ivan Lozano806efd32024-12-11 21:38:53 +00002138 } else if library, ok := ctx.Module().(LinkableInterface); ok && (library.CcLibraryInterface()) {
Ivan Lozano2b262972019-11-21 12:30:50 -08002139 // Non-cc.Modules may need an empty variant for their mutators.
2140 variations := []string{}
2141 if library.NonCcVariants() {
2142 variations = append(variations, "")
2143 }
Colin Cross127bb8b2020-12-16 16:46:01 -08002144 isLLNDK := false
Colin Cross767819f2024-05-22 14:22:34 -07002145 if m, ok := ctx.Module().(*Module); ok {
Colin Cross203b4212021-04-26 17:19:41 -07002146 isLLNDK = m.IsLlndk()
Colin Cross127bb8b2020-12-16 16:46:01 -08002147 }
2148 buildStatic := library.BuildStaticVariant() && !isLLNDK
2149 buildShared := library.BuildSharedVariant()
2150 if buildStatic && buildShared {
Colin Cross767819f2024-05-22 14:22:34 -07002151 variations = append([]string{"static", "shared"}, variations...)
2152 return variations
Colin Cross127bb8b2020-12-16 16:46:01 -08002153 } else if buildStatic {
Colin Cross767819f2024-05-22 14:22:34 -07002154 variations = append([]string{"static"}, variations...)
Colin Cross127bb8b2020-12-16 16:46:01 -08002155 } else if buildShared {
Colin Cross767819f2024-05-22 14:22:34 -07002156 variations = append([]string{"shared"}, variations...)
Ivan Lozano2b262972019-11-21 12:30:50 -08002157 }
Colin Cross767819f2024-05-22 14:22:34 -07002158
2159 if len(variations) > 0 {
2160 return variations
2161 }
2162 }
2163 return []string{""}
2164}
2165
2166func (linkageTransitionMutator) OutgoingTransition(ctx android.OutgoingTransitionContext, sourceVariation string) string {
2167 return ""
2168}
2169
2170func (linkageTransitionMutator) IncomingTransition(ctx android.IncomingTransitionContext, incomingVariation string) string {
2171 ccPrebuilt := false
2172 if m, ok := ctx.Module().(*Module); ok && m.linker != nil {
2173 _, ccPrebuilt = m.linker.(prebuiltLibraryInterface)
2174 }
2175 if ccPrebuilt {
2176 if incomingVariation != "" {
2177 return incomingVariation
2178 }
2179 library := ctx.Module().(*Module).linker.(prebuiltLibraryInterface)
2180 if library.buildShared() {
2181 return "shared"
2182 } else if library.buildStatic() {
2183 return "static"
2184 }
2185 return ""
2186 } else if library, ok := ctx.Module().(LinkableInterface); ok && library.CcLibraryInterface() {
2187 isLLNDK := false
2188 if m, ok := ctx.Module().(*Module); ok {
2189 isLLNDK = m.IsLlndk()
2190 }
2191 buildStatic := library.BuildStaticVariant() && !isLLNDK
2192 buildShared := library.BuildSharedVariant()
Ivan Lozano806efd32024-12-11 21:38:53 +00002193 if library.BuildRlibVariant() && !buildStatic && (incomingVariation == "static" || incomingVariation == "") {
Ivan Lozanofd47b1a2024-05-17 14:13:41 -04002194 // Rust modules do not build static libs, but rlibs are used as if they
2195 // were via `static_libs`. Thus we need to alias the BuildRlibVariant
2196 // to "static" for Rust FFI libraries.
Colin Cross767819f2024-05-22 14:22:34 -07002197 return ""
2198 }
2199 if incomingVariation != "" {
2200 return incomingVariation
2201 }
2202 if buildShared {
2203 return "shared"
2204 } else if buildStatic {
2205 return "static"
2206 }
2207 return ""
2208 }
2209 return ""
2210}
2211
2212func (linkageTransitionMutator) Mutate(ctx android.BottomUpMutatorContext, variation string) {
2213 ccPrebuilt := false
2214 if m, ok := ctx.Module().(*Module); ok && m.linker != nil {
2215 _, ccPrebuilt = m.linker.(prebuiltLibraryInterface)
2216 }
2217 if ccPrebuilt {
2218 library := ctx.Module().(*Module).linker.(prebuiltLibraryInterface)
2219 if variation == "static" {
2220 library.setStatic()
2221 if !library.buildStatic() {
2222 library.disablePrebuilt()
2223 }
2224 } else if variation == "shared" {
2225 library.setShared()
2226 if !library.buildShared() {
2227 library.disablePrebuilt()
2228 }
2229 }
2230 } else if library, ok := ctx.Module().(LinkableInterface); ok && library.CcLibraryInterface() {
2231 if variation == "static" {
2232 library.SetStatic()
2233 } else if variation == "shared" {
2234 library.SetShared()
2235 var isLLNDK bool
2236 if m, ok := ctx.Module().(*Module); ok {
2237 isLLNDK = m.IsLlndk()
2238 }
2239 buildStatic := library.BuildStaticVariant() && !isLLNDK
2240 buildShared := library.BuildSharedVariant()
2241 if buildStatic && buildShared {
2242 if _, ok := library.(*Module); ok {
2243 reuseStaticLibrary(ctx, library.(*Module))
2244 }
2245 }
Ivan Lozanofd47b1a2024-05-17 14:13:41 -04002246 }
Colin Crossb916a382016-07-29 17:28:03 -07002247 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07002248}
Jiyong Park7ed9de32018-10-15 22:25:07 +09002249
Chris Parsons3c27ca32020-11-20 12:42:07 -05002250// normalizeVersions modifies `versions` in place, so that each raw version
2251// string becomes its normalized canonical form.
2252// Validates that the versions in `versions` are specified in least to greatest order.
Colin Cross8ff10582023-12-07 13:10:56 -08002253func normalizeVersions(ctx android.BaseModuleContext, versions []string) {
Dan Albertc8060532020-07-22 22:32:17 -07002254 var previous android.ApiLevel
Jooyung Hanad4c1872020-02-27 17:56:44 +09002255 for i, v := range versions {
Dan Albertc8060532020-07-22 22:32:17 -07002256 ver, err := android.ApiLevelFromUser(ctx, v)
Jooyung Hanad4c1872020-02-27 17:56:44 +09002257 if err != nil {
Jooyung Hanaed150d2020-04-02 01:41:41 +09002258 ctx.PropertyErrorf("versions", "%s", err.Error())
2259 return
Jooyung Hanad4c1872020-02-27 17:56:44 +09002260 }
Dan Albertc8060532020-07-22 22:32:17 -07002261 if i > 0 && ver.LessThanOrEqualTo(previous) {
2262 ctx.PropertyErrorf("versions", "not sorted: %v", versions)
2263 }
2264 versions[i] = ver.String()
2265 previous = ver
Jooyung Hanaed150d2020-04-02 01:41:41 +09002266 }
Jooyung Hanad4c1872020-02-27 17:56:44 +09002267}
2268
Colin Crossadd04a82024-05-22 09:57:59 -07002269func perApiVersionVariations(mctx android.BaseModuleContext, minSdkVersion string) []string {
Colin Crossbbc941b2020-09-30 12:27:01 -07002270 from, err := nativeApiLevelFromUser(mctx, minSdkVersion)
2271 if err != nil {
2272 mctx.PropertyErrorf("min_sdk_version", err.Error())
Colin Crossadd04a82024-05-22 09:57:59 -07002273 return []string{""}
Colin Crossbbc941b2020-09-30 12:27:01 -07002274 }
2275
Colin Crossadd04a82024-05-22 09:57:59 -07002276 return ndkLibraryVersions(mctx, from)
Colin Crossbbc941b2020-09-30 12:27:01 -07002277}
2278
Liz Kammer23942242022-04-08 15:41:00 -04002279func canBeOrLinkAgainstVersionVariants(module interface {
Jooyung Han624d35c2020-04-10 12:57:24 +09002280 Host() bool
2281 InRamdisk() bool
Yifan Hong60e0cfb2020-10-21 15:17:56 -07002282 InVendorRamdisk() bool
Jooyung Han624d35c2020-04-10 12:57:24 +09002283}) bool {
Jose Galmes6f843bc2020-12-11 13:36:29 -08002284 return !module.Host() && !module.InRamdisk() && !module.InVendorRamdisk()
Jooyung Han624d35c2020-04-10 12:57:24 +09002285}
2286
Liz Kammer23942242022-04-08 15:41:00 -04002287func canBeVersionVariant(module interface {
Colin Cross3146c5c2020-09-30 15:34:40 -07002288 Host() bool
2289 InRamdisk() bool
Yifan Hong60e0cfb2020-10-21 15:17:56 -07002290 InVendorRamdisk() bool
Colin Cross3146c5c2020-09-30 15:34:40 -07002291 CcLibraryInterface() bool
2292 Shared() bool
Colin Cross3146c5c2020-09-30 15:34:40 -07002293}) bool {
Liz Kammer23942242022-04-08 15:41:00 -04002294 return canBeOrLinkAgainstVersionVariants(module) &&
Colin Crossa717db72020-10-23 14:53:06 -07002295 module.CcLibraryInterface() && module.Shared()
Colin Cross3146c5c2020-09-30 15:34:40 -07002296}
2297
Colin Cross127bb8b2020-12-16 16:46:01 -08002298func moduleLibraryInterface(module blueprint.Module) libraryInterface {
Colin Cross31076b32020-10-23 17:22:06 -07002299 if m, ok := module.(*Module); ok {
2300 return m.library
2301 }
2302 return nil
2303}
2304
Liz Kammer23942242022-04-08 15:41:00 -04002305// setStubsVersions normalizes the versions in the Stubs.Versions property into MutatedProperties.AllStubsVersions.
Colin Crossadd04a82024-05-22 09:57:59 -07002306func setStubsVersions(mctx android.BaseModuleContext, library libraryInterface, module *Module) {
Liz Kammer23942242022-04-08 15:41:00 -04002307 if !library.buildShared() || !canBeVersionVariant(module) {
2308 return
Colin Crossd1f898e2020-08-18 18:35:15 -07002309 }
Liz Kammer23942242022-04-08 15:41:00 -04002310 versions := library.stubsVersions(mctx)
Liz Kammer23942242022-04-08 15:41:00 -04002311 if mctx.Failed() {
2312 return
2313 }
2314 // Set the versions on the pre-mutated module so they can be read by any llndk modules that
2315 // depend on the implementation library and haven't been mutated yet.
2316 library.setAllStubsVersions(versions)
Colin Crossd1f898e2020-08-18 18:35:15 -07002317}
Jooyung Han61b66e92020-03-21 14:21:46 +00002318
Colin Crossadd04a82024-05-22 09:57:59 -07002319// versionTransitionMutator splits a module into the mandatory non-stubs variant
Colin Crossd1f898e2020-08-18 18:35:15 -07002320// (which is unnamed) and zero or more stubs variants.
Colin Crossadd04a82024-05-22 09:57:59 -07002321type versionTransitionMutator struct{}
2322
2323func (versionTransitionMutator) Split(ctx android.BaseModuleContext) []string {
2324 if ctx.Os() != android.Android {
2325 return []string{""}
Liz Kammer23942242022-04-08 15:41:00 -04002326 }
2327
Colin Crossadd04a82024-05-22 09:57:59 -07002328 m, ok := ctx.Module().(*Module)
2329 if library := moduleLibraryInterface(ctx.Module()); library != nil && canBeVersionVariant(m) {
2330 setStubsVersions(ctx, library, m)
Liz Kammer23942242022-04-08 15:41:00 -04002331
Colin Crossadd04a82024-05-22 09:57:59 -07002332 return append(slices.Clone(library.allStubsVersions()), "")
2333 } else if ok && m.SplitPerApiLevel() && m.IsSdkVariant() {
2334 return perApiVersionVariations(ctx, m.MinSdkVersion())
Colin Crossbbc941b2020-09-30 12:27:01 -07002335 }
2336
Colin Crossadd04a82024-05-22 09:57:59 -07002337 return []string{""}
2338}
2339
2340func (versionTransitionMutator) OutgoingTransition(ctx android.OutgoingTransitionContext, sourceVariation string) string {
2341 return ""
2342}
2343
2344func (versionTransitionMutator) IncomingTransition(ctx android.IncomingTransitionContext, incomingVariation string) string {
2345 if ctx.Os() != android.Android {
2346 return ""
2347 }
2348 m, ok := ctx.Module().(*Module)
2349 if library := moduleLibraryInterface(ctx.Module()); library != nil && canBeVersionVariant(m) {
2350 if incomingVariation == "latest" {
2351 latestVersion := ""
2352 versions := library.allStubsVersions()
2353 if len(versions) > 0 {
2354 latestVersion = versions[len(versions)-1]
2355 }
2356 return latestVersion
Colin Crossbbc941b2020-09-30 12:27:01 -07002357 }
Colin Crossadd04a82024-05-22 09:57:59 -07002358 return incomingVariation
2359 } else if ok && m.SplitPerApiLevel() && m.IsSdkVariant() {
2360 // If this module only has variants with versions and the incoming dependency doesn't specify which one
2361 // is needed then assume the latest version.
2362 if incomingVariation == "" {
2363 return android.FutureApiLevel.String()
2364 }
2365 return incomingVariation
2366 }
2367
2368 return ""
2369}
2370
2371func (versionTransitionMutator) Mutate(ctx android.BottomUpMutatorContext, variation string) {
2372 // Optimization: return early if this module can't be affected.
2373 if ctx.Os() != android.Android {
2374 return
2375 }
2376
2377 m, ok := ctx.Module().(*Module)
2378 if library := moduleLibraryInterface(ctx.Module()); library != nil && canBeVersionVariant(m) {
2379 isLLNDK := m.IsLlndk()
2380 isVendorPublicLibrary := m.IsVendorPublicLibrary()
Colin Crossadd04a82024-05-22 09:57:59 -07002381
Spandan Dasff665182024-09-11 18:48:44 +00002382 if variation != "" || isLLNDK || isVendorPublicLibrary {
Colin Crossadd04a82024-05-22 09:57:59 -07002383 // A stubs or LLNDK stubs variant.
2384 if m.sanitize != nil {
2385 m.sanitize.Properties.ForceDisable = true
2386 }
2387 if m.stl != nil {
2388 m.stl.Properties.Stl = StringPtr("none")
2389 }
2390 m.Properties.PreventInstall = true
2391 lib := moduleLibraryInterface(m)
2392 allStubsVersions := library.allStubsVersions()
2393 isLatest := len(allStubsVersions) > 0 && variation == allStubsVersions[len(allStubsVersions)-1]
2394 lib.setBuildStubs(isLatest)
2395 }
2396 if variation != "" {
2397 // A non-LLNDK stubs module is hidden from make
2398 library.setStubsVersion(variation)
2399 m.Properties.HideFromMake = true
2400 } else {
2401 // A non-LLNDK implementation module has a dependency to all stubs versions
2402 for _, version := range library.allStubsVersions() {
2403 ctx.AddVariationDependencies([]blueprint.Variation{{"version", version}},
2404 stubImplDepTag, ctx.ModuleName())
2405 }
2406 }
2407 } else if ok && m.SplitPerApiLevel() && m.IsSdkVariant() {
2408 m.Properties.Sdk_version = StringPtr(variation)
2409 m.Properties.Min_sdk_version = StringPtr(variation)
Jiyong Park7ed9de32018-10-15 22:25:07 +09002410 }
Jiyong Park7ed9de32018-10-15 22:25:07 +09002411}
Colin Crossd7227f92019-09-05 14:26:33 -07002412
2413// maybeInjectBoringSSLHash adds a rule to run bssl_inject_hash on the output file if the module has the
2414// inject_bssl_hash or if any static library dependencies have inject_bssl_hash set. It returns the output path
2415// that the linked output file should be written to.
2416// TODO(b/137267623): Remove this in favor of a cc_genrule when they support operating on shared libraries.
2417func maybeInjectBoringSSLHash(ctx android.ModuleContext, outputFile android.ModuleOutPath,
2418 inject *bool, fileName string) android.ModuleOutPath {
2419 // TODO(b/137267623): Remove this in favor of a cc_genrule when they support operating on shared libraries.
2420 injectBoringSSLHash := Bool(inject)
2421 ctx.VisitDirectDeps(func(dep android.Module) {
Colin Cross6e511a92020-07-27 21:26:48 -07002422 if tag, ok := ctx.OtherModuleDependencyTag(dep).(libraryDependencyTag); ok && tag.static() {
Colin Crossd7227f92019-09-05 14:26:33 -07002423 if cc, ok := dep.(*Module); ok {
2424 if library, ok := cc.linker.(*libraryDecorator); ok {
2425 if Bool(library.Properties.Inject_bssl_hash) {
2426 injectBoringSSLHash = true
2427 }
2428 }
2429 }
2430 }
2431 })
2432 if injectBoringSSLHash {
2433 hashedOutputfile := outputFile
2434 outputFile = android.PathForModuleOut(ctx, "unhashed", fileName)
2435
Colin Crossf1a035e2020-11-16 17:32:30 -08002436 rule := android.NewRuleBuilder(pctx, ctx)
Colin Crossd7227f92019-09-05 14:26:33 -07002437 rule.Command().
Colin Crossf1a035e2020-11-16 17:32:30 -08002438 BuiltTool("bssl_inject_hash").
Colin Crossd7227f92019-09-05 14:26:33 -07002439 FlagWithInput("-in-object ", outputFile).
2440 FlagWithOutput("-o ", hashedOutputfile)
Colin Crossf1a035e2020-11-16 17:32:30 -08002441 rule.Build("injectCryptoHash", "inject crypto hash")
Colin Crossd7227f92019-09-05 14:26:33 -07002442 }
2443
2444 return outputFile
2445}