blob: 7b854864f767c9a4add276ba59a5125f726c80e6 [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"
Ivan Lozano18ac30a2024-10-30 18:04:06 +000028 "android/soong/cc/config"
Kiyoung Kimaa394802024-01-08 12:55:45 +090029
Chris Parsons51f8c392021-08-03 21:01:05 -040030 "github.com/google/blueprint"
Colin Crossa14fb6a2024-10-23 16:57:06 -070031 "github.com/google/blueprint/depset"
Chris Parsons51f8c392021-08-03 21:01:05 -040032 "github.com/google/blueprint/pathtools"
Spandan Das4238c652022-09-09 01:38:47 +000033 "github.com/google/blueprint/proptools"
Colin Cross4d9c2d12016-07-29 12:48:20 -070034)
35
Liz Kammer2b376bc2022-01-12 12:00:49 -050036// LibraryProperties is a collection of properties shared by cc library rules/cc.
Colin Crossb916a382016-07-29 17:28:03 -070037type LibraryProperties struct {
Jerome Gaillard9b454732024-12-10 18:07:19 +000038 // local file name to pass to the linker as -exported_symbols_list
39 Exported_symbols_list *string `android:"path,arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070040 // local file name to pass to the linker as -unexported_symbols_list
Colin Cross27b922f2019-03-04 22:35:41 -080041 Unexported_symbols_list *string `android:"path,arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070042 // local file name to pass to the linker as -force_symbols_not_weak_list
Colin Cross27b922f2019-03-04 22:35:41 -080043 Force_symbols_not_weak_list *string `android:"path,arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070044 // local file name to pass to the linker as -force_symbols_weak_list
Colin Cross27b922f2019-03-04 22:35:41 -080045 Force_symbols_weak_list *string `android:"path,arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070046
47 // rename host libraries to prevent overlap with system installed libraries
48 Unique_host_soname *bool
49
Dan Willemsene1240db2016-11-03 14:28:51 -070050 Aidl struct {
51 // export headers generated from .aidl sources
Nan Zhang0007d812017-11-07 10:57:05 -080052 Export_aidl_headers *bool
Dan Willemsene1240db2016-11-03 14:28:51 -070053 }
54
Colin Cross0c461f12016-10-20 16:11:43 -070055 Proto struct {
56 // export headers generated from .proto sources
Nan Zhang0007d812017-11-07 10:57:05 -080057 Export_proto_headers *bool
Colin Cross0c461f12016-10-20 16:11:43 -070058 }
Dan Albertf563d252017-10-13 00:29:00 -070059
Inseob Kimc0907f12019-02-08 21:00:45 +090060 Sysprop struct {
61 // Whether platform owns this sysprop library.
62 Platform *bool
Inseob Kimb3f22ca2019-03-05 12:40:24 +090063 } `blueprint:"mutated"`
Inseob Kimc0907f12019-02-08 21:00:45 +090064
Nan Zhang0007d812017-11-07 10:57:05 -080065 Static_ndk_lib *bool
Jiyong Park7ed9de32018-10-15 22:25:07 +090066
Florian Mayer6dab96c2020-12-31 11:39:48 +000067 // Generate stubs to make this library accessible to APEXes.
Ivan Lozano9eaacc82024-10-30 14:28:17 +000068 Stubs StubsProperties `android:"arch_variant"`
dimitryd95964a2018-11-07 13:43:34 +010069
70 // set the name of the output
71 Stem *string `android:"arch_variant"`
72
Colin Cross0fd6a412019-08-16 14:22:10 -070073 // set suffix of the name of the output
74 Suffix *string `android:"arch_variant"`
75
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +080076 // Properties for ABI compatibility checker.
77 Header_abi_checker headerAbiCheckerProperties
78
Colin Cross0fd6a412019-08-16 14:22:10 -070079 Target struct {
Justin Yun63e9ec72020-10-29 16:49:43 +090080 Vendor, Product struct {
Colin Cross0fd6a412019-08-16 14:22:10 -070081 // set suffix of the name of the output
82 Suffix *string `android:"arch_variant"`
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +080083
84 Header_abi_checker headerAbiCheckerProperties
Jooyung Han85707de2023-12-01 14:21:13 +090085
86 // Disable stubs for vendor/product variants
87 // This is a workaround to keep `stubs` only for "core" variant (not product/vendor).
88 // It would be nice if we could put `stubs` into a `target: { core: {} }`
89 // block but it's not supported in soong yet. This could be removed/simplified once we have
90 // a better syntax.
91 No_stubs bool
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +080092 }
93
94 Platform struct {
95 Header_abi_checker headerAbiCheckerProperties
Colin Cross0fd6a412019-08-16 14:22:10 -070096 }
97 }
98
dimitryd95964a2018-11-07 13:43:34 +010099 // Names of modules to be overridden. Listed modules can only be other shared libraries
100 // (in Make or Soong).
101 // This does not completely prevent installation of the overridden libraries, but if both
102 // binaries would be installed by default (in PRODUCT_PACKAGES) the other library will be removed
103 // from PRODUCT_PACKAGES.
104 Overrides []string
Logan Chiene3d7a0d2019-01-17 00:18:02 +0800105
Pete Bentley803e1612019-08-06 22:19:59 +0100106 // Inject boringssl hash into the shared library. This is only intended for use by external/boringssl.
107 Inject_bssl_hash *bool `android:"arch_variant"`
Colin Cross0477b422020-10-13 18:43:54 -0700108
Colin Cross127bb8b2020-12-16 16:46:01 -0800109 // If this is an LLNDK library, properties to describe the LLNDK stubs. Will be copied from
110 // the module pointed to by llndk_stubs if it is set.
Spandan Das02be1012024-07-24 01:21:53 +0000111 Llndk llndkLibraryProperties `android:"arch_variant"`
Colin Cross5271fea2021-04-27 13:06:04 -0700112
113 // If this is a vendor public library, properties to describe the vendor public library stubs.
114 Vendor_public_library vendorPublicLibraryProperties
Colin Crossa48ab5b2017-02-14 15:28:44 -0800115}
Colin Cross0c461f12016-10-20 16:11:43 -0700116
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000117type StubsProperties struct {
118 // Relative path to the symbol map. The symbol map provides the list of
119 // symbols that are exported for stubs variant of this library.
120 Symbol_file *string `android:"path,arch_variant"`
121
122 // List versions to generate stubs libs for. The version name "current" is always
123 // implicitly added.
124 Versions []string
125
126 // Whether to not require the implementation of the library to be installed if a
127 // client of the stubs is installed. Defaults to true; set to false if the
128 // implementation is made available by some other means, e.g. in a Microdroid
129 // virtual machine.
130 Implementation_installable *bool
131}
132
Chris Parsons3c27ca32020-11-20 12:42:07 -0500133// StaticProperties is a properties stanza to affect only attributes of the "static" variants of a
134// library module.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700135type StaticProperties struct {
136 Static StaticOrSharedProperties `android:"arch_variant"`
137}
138
Chris Parsons3c27ca32020-11-20 12:42:07 -0500139// SharedProperties is a properties stanza to affect only attributes of the "shared" variants of a
140// library module.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700141type SharedProperties struct {
142 Shared StaticOrSharedProperties `android:"arch_variant"`
143}
144
Chris Parsons3c27ca32020-11-20 12:42:07 -0500145// StaticOrSharedProperties is an embedded struct representing properties to affect attributes of
146// either only the "static" variants or only the "shared" variants of a library module. These override
147// the base properties of the same name.
148// Use `StaticProperties` or `SharedProperties`, depending on which variant is needed.
149// `StaticOrSharedProperties` exists only to avoid duplication.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700150type StaticOrSharedProperties struct {
Cole Faust96a692b2024-08-08 14:47:51 -0700151 Srcs proptools.Configurable[[]string] `android:"path,arch_variant"`
Evgenii Stepanov2080bfe2020-07-24 15:35:40 -0700152
Chih-Hung Hsieh769a51c2021-09-17 17:18:39 -0700153 Tidy_disabled_srcs []string `android:"path,arch_variant"`
154
Chih-Hung Hsieh9db8a0c2022-02-17 12:54:45 -0800155 Tidy_timeout_srcs []string `android:"path,arch_variant"`
156
Evgenii Stepanov2080bfe2020-07-24 15:35:40 -0700157 Sanitized Sanitized `android:"arch_variant"`
158
Cole Fauste96c16a2024-06-13 14:51:14 -0700159 Cflags proptools.Configurable[[]string] `android:"arch_variant"`
Colin Crosse1bb5d02019-09-24 14:55:04 -0700160
Cole Faustf0006e72024-08-19 14:39:19 -0700161 Enabled *bool `android:"arch_variant"`
162 Whole_static_libs proptools.Configurable[[]string] `android:"arch_variant"`
163 Static_libs proptools.Configurable[[]string] `android:"arch_variant"`
164 Shared_libs proptools.Configurable[[]string] `android:"arch_variant"`
165 System_shared_libs []string `android:"arch_variant"`
Colin Crosse1bb5d02019-09-24 14:55:04 -0700166
167 Export_shared_lib_headers []string `android:"arch_variant"`
168 Export_static_lib_headers []string `android:"arch_variant"`
Jiyong Parka90ca002019-10-07 15:47:24 +0900169
170 Apex_available []string `android:"arch_variant"`
Colin Cross1bc94122021-10-28 13:25:54 -0700171
172 Installable *bool `android:"arch_variant"`
Colin Crosse1bb5d02019-09-24 14:55:04 -0700173}
174
Colin Crossa48ab5b2017-02-14 15:28:44 -0800175type LibraryMutatedProperties struct {
Colin Crossb916a382016-07-29 17:28:03 -0700176 // Build a static variant
177 BuildStatic bool `blueprint:"mutated"`
178 // Build a shared variant
179 BuildShared bool `blueprint:"mutated"`
180 // This variant is shared
181 VariantIsShared bool `blueprint:"mutated"`
182 // This variant is static
183 VariantIsStatic bool `blueprint:"mutated"`
Jiyong Park7ed9de32018-10-15 22:25:07 +0900184
185 // This variant is a stubs lib
186 BuildStubs bool `blueprint:"mutated"`
Jiyong Parkd4a3a132021-03-17 20:21:35 +0900187 // This variant is the latest version
188 IsLatestVersion bool `blueprint:"mutated"`
Jiyong Park7ed9de32018-10-15 22:25:07 +0900189 // Version of the stubs lib
190 StubsVersion string `blueprint:"mutated"`
Colin Crossd1f898e2020-08-18 18:35:15 -0700191 // List of all stubs versions associated with an implementation lib
192 AllStubsVersions []string `blueprint:"mutated"`
Colin Crossb916a382016-07-29 17:28:03 -0700193}
194
195type FlagExporterProperties struct {
196 // list of directories relative to the Blueprints file that will
Dan Willemsen273af7f2016-11-03 15:53:42 -0700197 // be added to the include path (using -I) for this module and any module that links
Colin Cross5d195602017-10-17 16:15:50 -0700198 // against this module. Directories listed in export_include_dirs do not need to be
199 // listed in local_include_dirs.
Aleks Todorovc9becde2024-06-10 12:51:53 +0100200 Export_include_dirs proptools.Configurable[[]string] `android:"arch_variant,variant_prepend"`
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700201
Jiyong Park73c54ee2019-10-22 20:31:18 +0900202 // list of directories that will be added to the system include path
203 // using -isystem for this module and any module that links against this module.
Colin Cross0ed579e2021-06-29 00:51:12 +0000204 Export_system_include_dirs []string `android:"arch_variant,variant_prepend"`
Jiyong Park73c54ee2019-10-22 20:31:18 +0900205
Alessandro Astone10e9d432020-10-01 12:59:27 +0200206 // list of plain cc flags to be used for any module that links against this module.
Michael Bestasb3275bc2025-04-03 13:06:08 +0300207 Export_cflags proptools.Configurable[[]string] `android:"arch_variant"`
Alessandro Astone10e9d432020-10-01 12:59:27 +0200208
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700209 Target struct {
Justin Yun63e9ec72020-10-29 16:49:43 +0900210 Vendor, Product struct {
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700211 // list of exported include directories, like
Justin Yun63e9ec72020-10-29 16:49:43 +0900212 // export_include_dirs, that will be applied to
213 // vendor or product variant of this library.
214 // This will overwrite any other declarations.
Steven Morelandb21df8f2018-01-05 14:42:54 -0800215 Override_export_include_dirs []string
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700216 }
217 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700218}
219
220func init() {
Paul Duffin6c26dc72019-12-19 15:02:40 +0000221 RegisterLibraryBuildComponents(android.InitRegistrationContext)
222}
223
224func RegisterLibraryBuildComponents(ctx android.RegistrationContext) {
225 ctx.RegisterModuleType("cc_library_static", LibraryStaticFactory)
Ivan Lozano2afdfe62025-02-27 04:19:23 -0800226 ctx.RegisterModuleType("cc_rustlibs_for_make", LibraryMakeRustlibsFactory)
Paul Duffin6c26dc72019-12-19 15:02:40 +0000227 ctx.RegisterModuleType("cc_library_shared", LibrarySharedFactory)
228 ctx.RegisterModuleType("cc_library", LibraryFactory)
229 ctx.RegisterModuleType("cc_library_host_static", LibraryHostStaticFactory)
230 ctx.RegisterModuleType("cc_library_host_shared", LibraryHostSharedFactory)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700231}
232
Patrice Arruda83c89e02019-03-25 15:32:39 -0700233// cc_library creates both static and/or shared libraries for a device and/or
234// host. By default, a cc_library has a single variant that targets the device.
235// Specifying `host_supported: true` also creates a library that targets the
236// host.
Steven Morelandf9e62162017-11-02 17:00:50 -0700237func LibraryFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800238 module, _ := NewLibrary(android.HostAndDeviceSupported)
Paul Duffina0843f62019-12-13 19:50:38 +0000239 // Can be used as both a static and a shared library.
240 module.sdkMemberTypes = []android.SdkMemberType{
241 sharedLibrarySdkMemberType,
242 staticLibrarySdkMemberType,
Paul Duffin9b76c0b2020-03-12 10:24:35 +0000243 staticAndSharedLibrarySdkMemberType,
Paul Duffina0843f62019-12-13 19:50:38 +0000244 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700245 return module.Init()
246}
247
Patrice Arruda83c89e02019-03-25 15:32:39 -0700248// cc_library_static creates a static library for a device and/or host binary.
Steven Morelandf9e62162017-11-02 17:00:50 -0700249func LibraryStaticFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800250 module, library := NewLibrary(android.HostAndDeviceSupported)
251 library.BuildOnlyStatic()
Paul Duffina0843f62019-12-13 19:50:38 +0000252 module.sdkMemberTypes = []android.SdkMemberType{staticLibrarySdkMemberType}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700253 return module.Init()
254}
255
Ivan Lozano2afdfe62025-02-27 04:19:23 -0800256// cc_rustlibs_for_make creates a static library which bundles together rust_ffi_static
257// deps for Make. This should not be depended on in Soong, and is probably not the
258// module you need unless you are sure of what you're doing. These should only
259// be declared as dependencies in Make. To ensure inclusion, rust_ffi_static modules
260// should be declared in the whole_static_libs property.
261func LibraryMakeRustlibsFactory() android.Module {
262 module, library := NewLibrary(android.HostAndDeviceSupported)
263 library.BuildOnlyStatic()
264 library.wideStaticlibForMake = true
265 module.sdkMemberTypes = []android.SdkMemberType{staticLibrarySdkMemberType}
266 return module.Init()
267}
268
Patrice Arruda83c89e02019-03-25 15:32:39 -0700269// cc_library_shared creates a shared library for a device and/or host.
Steven Morelandf9e62162017-11-02 17:00:50 -0700270func LibrarySharedFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800271 module, library := NewLibrary(android.HostAndDeviceSupported)
272 library.BuildOnlyShared()
Paul Duffina0843f62019-12-13 19:50:38 +0000273 module.sdkMemberTypes = []android.SdkMemberType{sharedLibrarySdkMemberType}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700274 return module.Init()
275}
276
Patrice Arruda83c89e02019-03-25 15:32:39 -0700277// cc_library_host_static creates a static library that is linkable to a host
278// binary.
Steven Morelandf9e62162017-11-02 17:00:50 -0700279func LibraryHostStaticFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800280 module, library := NewLibrary(android.HostSupported)
281 library.BuildOnlyStatic()
Paul Duffina0843f62019-12-13 19:50:38 +0000282 module.sdkMemberTypes = []android.SdkMemberType{staticLibrarySdkMemberType}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700283 return module.Init()
284}
285
Patrice Arruda83c89e02019-03-25 15:32:39 -0700286// cc_library_host_shared creates a shared library that is usable on a host.
Steven Morelandf9e62162017-11-02 17:00:50 -0700287func LibraryHostSharedFactory() android.Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800288 module, library := NewLibrary(android.HostSupported)
289 library.BuildOnlyShared()
Paul Duffina0843f62019-12-13 19:50:38 +0000290 module.sdkMemberTypes = []android.SdkMemberType{sharedLibrarySdkMemberType}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700291 return module.Init()
292}
293
Chris Parsons3c27ca32020-11-20 12:42:07 -0500294// flagExporter is a separated portion of libraryDecorator pertaining to exported
295// include paths and flags. Keeping this dependency-related information separate
296// from the rest of library information is helpful in keeping data more structured
297// and explicit.
Colin Cross4d9c2d12016-07-29 12:48:20 -0700298type flagExporter struct {
299 Properties FlagExporterProperties
300
Ivan Lozano0a468a42024-05-13 21:03:34 -0400301 dirs android.Paths // Include directories to be included with -I
302 systemDirs android.Paths // System include directories to be included with -isystem
303 flags []string // Exported raw flags.
304 deps android.Paths
305 headers android.Paths
306 rustRlibDeps []RustRlibDep
Colin Cross4d9c2d12016-07-29 12:48:20 -0700307}
308
Chris Parsons3c27ca32020-11-20 12:42:07 -0500309// exportedIncludes returns the effective include paths for this module and
310// any module that links against this module. This is obtained from
311// the export_include_dirs property in the appropriate target stanza.
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700312func (f *flagExporter) exportedIncludes(ctx ModuleContext) android.Paths {
Justin Yun6977e8a2020-10-29 18:24:11 +0900313 if ctx.inVendor() && f.Properties.Target.Vendor.Override_export_include_dirs != nil {
Steven Morelandb21df8f2018-01-05 14:42:54 -0800314 return android.PathsForModuleSrc(ctx, f.Properties.Target.Vendor.Override_export_include_dirs)
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700315 }
Justin Yun6977e8a2020-10-29 18:24:11 +0900316 if ctx.inProduct() && f.Properties.Target.Product.Override_export_include_dirs != nil {
317 return android.PathsForModuleSrc(ctx, f.Properties.Target.Product.Override_export_include_dirs)
318 }
Aleks Todorovc9becde2024-06-10 12:51:53 +0100319 return android.PathsForModuleSrc(ctx, f.Properties.Export_include_dirs.GetOrDefault(ctx, nil))
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700320}
321
Hsin-Yi Chen5f228b02024-04-02 12:38:47 +0800322func (f *flagExporter) exportedSystemIncludes(ctx ModuleContext) android.Paths {
323 return android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)
324}
325
Chris Parsons3c27ca32020-11-20 12:42:07 -0500326// exportIncludes registers the include directories and system include directories to be exported
327// transitively to modules depending on this module.
Inseob Kim69378442019-06-03 19:10:47 +0900328func (f *flagExporter) exportIncludes(ctx ModuleContext) {
Jiyong Park74955042019-10-22 20:19:51 +0900329 f.dirs = append(f.dirs, f.exportedIncludes(ctx)...)
Jiyong Park73c54ee2019-10-22 20:31:18 +0900330 f.systemDirs = append(f.systemDirs, android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700331}
332
Alessandro Astone10e9d432020-10-01 12:59:27 +0200333func (f *flagExporter) exportExtraFlags(ctx ModuleContext) {
Michael Bestasb3275bc2025-04-03 13:06:08 +0300334 f.flags = append(f.flags, f.Properties.Export_cflags.GetOrDefault(ctx, nil)...)
Alessandro Astone10e9d432020-10-01 12:59:27 +0200335}
336
Chris Parsons3c27ca32020-11-20 12:42:07 -0500337// exportIncludesAsSystem registers the include directories and system include directories to be
338// exported transitively both as system include directories to modules depending on this module.
Inseob Kim69378442019-06-03 19:10:47 +0900339func (f *flagExporter) exportIncludesAsSystem(ctx ModuleContext) {
Jiyong Park73c54ee2019-10-22 20:31:18 +0900340 // all dirs are force exported as system
Jiyong Park74955042019-10-22 20:19:51 +0900341 f.systemDirs = append(f.systemDirs, f.exportedIncludes(ctx)...)
Jiyong Park73c54ee2019-10-22 20:31:18 +0900342 f.systemDirs = append(f.systemDirs, android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)...)
Inseob Kim69378442019-06-03 19:10:47 +0900343}
344
Chris Parsons3c27ca32020-11-20 12:42:07 -0500345// reexportDirs registers the given directories as include directories to be exported transitively
346// to modules depending on this module.
Jiyong Park74955042019-10-22 20:19:51 +0900347func (f *flagExporter) reexportDirs(dirs ...android.Path) {
Inseob Kim69378442019-06-03 19:10:47 +0900348 f.dirs = append(f.dirs, dirs...)
349}
350
Chris Parsons3c27ca32020-11-20 12:42:07 -0500351// reexportSystemDirs registers the given directories as system include directories
352// to be exported transitively to modules depending on this module.
Jiyong Park74955042019-10-22 20:19:51 +0900353func (f *flagExporter) reexportSystemDirs(dirs ...android.Path) {
Inseob Kim69378442019-06-03 19:10:47 +0900354 f.systemDirs = append(f.systemDirs, dirs...)
355}
356
Chris Parsons3c27ca32020-11-20 12:42:07 -0500357// reexportFlags registers the flags to be exported transitively to modules depending on this
358// module.
Inseob Kim69378442019-06-03 19:10:47 +0900359func (f *flagExporter) reexportFlags(flags ...string) {
Jaewoong Jung3aff5782020-02-11 07:54:35 -0800360 if android.PrefixInList(flags, "-I") || android.PrefixInList(flags, "-isystem") {
361 panic(fmt.Errorf("Exporting invalid flag %q: "+
362 "use reexportDirs or reexportSystemDirs to export directories", flag))
Inseob Kim69378442019-06-03 19:10:47 +0900363 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700364 f.flags = append(f.flags, flags...)
365}
366
Inseob Kim69378442019-06-03 19:10:47 +0900367func (f *flagExporter) reexportDeps(deps ...android.Path) {
368 f.deps = append(f.deps, deps...)
369}
370
Ivan Lozano0a468a42024-05-13 21:03:34 -0400371func (f *flagExporter) reexportRustStaticDeps(deps ...RustRlibDep) {
372 f.rustRlibDeps = append(f.rustRlibDeps, deps...)
373}
374
Inseob Kimd110f872019-12-06 13:15:38 +0900375// addExportedGeneratedHeaders does nothing but collects generated header files.
376// This can be differ to exportedDeps which may contain phony files to minimize ninja.
377func (f *flagExporter) addExportedGeneratedHeaders(headers ...android.Path) {
378 f.headers = append(f.headers, headers...)
379}
380
Colin Cross0de8a1e2020-09-18 14:15:30 -0700381func (f *flagExporter) setProvider(ctx android.ModuleContext) {
Colin Cross40213022023-12-13 15:19:49 -0800382 android.SetProvider(ctx, FlagExporterInfoProvider, FlagExporterInfo{
Chris Parsonsf60ecf02021-04-27 14:48:30 -0400383 // Comes from Export_include_dirs property, and those of exported transitive deps
384 IncludeDirs: android.FirstUniquePaths(f.dirs),
385 // Comes from Export_system_include_dirs property, and those of exported transitive deps
Colin Crossc82e6e22021-04-20 18:21:50 -0700386 SystemIncludeDirs: android.FirstUniquePaths(f.systemDirs),
Chris Parsonsf60ecf02021-04-27 14:48:30 -0400387 // Used in very few places as a one-off way of adding extra defines.
388 Flags: f.flags,
389 // Used sparingly, for extra files that need to be explicitly exported to dependers,
390 // or for phony files to minimize ninja.
391 Deps: f.deps,
Ivan Lozano0a468a42024-05-13 21:03:34 -0400392 // Used for exporting rlib deps of static libraries to dependents.
393 RustRlibDeps: f.rustRlibDeps,
Chris Parsonsf60ecf02021-04-27 14:48:30 -0400394 // For exported generated headers, such as exported aidl headers, proto headers, or
395 // sysprop headers.
396 GeneratedHeaders: f.headers,
Colin Cross0de8a1e2020-09-18 14:15:30 -0700397 })
Inseob Kim69378442019-06-03 19:10:47 +0900398}
399
Colin Crossb916a382016-07-29 17:28:03 -0700400// libraryDecorator wraps baseCompiler, baseLinker and baseInstaller to provide library-specific
401// functionality: static vs. shared linkage, reusing object files for shared libraries
402type libraryDecorator struct {
Colin Crossa48ab5b2017-02-14 15:28:44 -0800403 Properties LibraryProperties
Colin Crosse1bb5d02019-09-24 14:55:04 -0700404 StaticProperties StaticProperties
405 SharedProperties SharedProperties
Colin Crossa48ab5b2017-02-14 15:28:44 -0800406 MutatedProperties LibraryMutatedProperties
Colin Cross4d9c2d12016-07-29 12:48:20 -0700407
408 // For reusing static library objects for shared library
Inseob Kim69378442019-06-03 19:10:47 +0900409 reuseObjects Objects
Colin Cross10d22312017-05-03 11:01:58 -0700410
Colin Cross26c34ed2016-09-30 17:10:16 -0700411 // table-of-contents file to optimize out relinking when possible
412 tocFile android.OptionalPath
Colin Cross4d9c2d12016-07-29 12:48:20 -0700413
Colin Cross4d9c2d12016-07-29 12:48:20 -0700414 flagExporter
Liz Kammerb6a55bf2021-04-12 15:42:51 -0400415 flagExporterInfo *FlagExporterInfo
416 stripper Stripper
Colin Cross4d9c2d12016-07-29 12:48:20 -0700417
Colin Cross4d9c2d12016-07-29 12:48:20 -0700418 // For whole_static_libs
Colin Crossa2bcf2c2022-02-11 13:11:55 -0800419 objects Objects
420 wholeStaticLibsFromPrebuilts android.Paths
Colin Cross4d9c2d12016-07-29 12:48:20 -0700421
422 // Uses the module's name if empty, but can be overridden. Does not include
423 // shlib suffix.
424 libName string
Colin Crossb916a382016-07-29 17:28:03 -0700425
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -0800426 sabi *sabi
427
Dan Willemsen581341d2017-02-09 16:16:31 -0800428 // Output archive of gcno coverage information files
429 coverageOutputFile android.OptionalPath
430
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -0800431 // Source Abi Diff
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +0800432 sAbiDiff android.Paths
Mu-Le Lee0a9005e2022-07-05 09:49:50 +0000433
Colin Cross0875c522017-11-28 17:34:01 -0800434 // Location of the static library in the sysroot. Empty if the library is
435 // not included in the NDK.
436 ndkSysrootPath android.Path
437
Colin Crossb60190a2018-09-04 16:28:17 -0700438 // Location of the linked, unstripped library for shared libraries
439 unstrippedOutputFile android.Path
Wei Li5f5d2712023-12-11 15:40:29 -0800440 // Location of the linked, stripped library for shared libraries, strip: "all"
441 strippedAllOutputFile android.Path
Colin Crossb60190a2018-09-04 16:28:17 -0700442
Cole Faustd143f3e2025-02-24 16:18:18 -0800443 // Location of the file that should be copied to dist dir when no explicit tag is requested
444 defaultDistFile android.Path
Dan Willemsen569edc52018-11-19 09:33:29 -0800445
Colin Cross8e21aa52020-09-28 18:28:02 -0700446 versionScriptPath android.OptionalPath
Jiyong Park7ed9de32018-10-15 22:25:07 +0900447
Jaewoong Jung18aefc12020-12-21 09:11:10 -0800448 postInstallCmds []string
Jiyong Parkf1194352019-02-25 11:05:47 +0900449
Colin Cross5ec407b2020-09-30 11:41:33 -0700450 skipAPIDefine bool
451
Martin Stjernholmc5dd4f72020-04-01 20:38:01 +0100452 // Decorated interfaces
Colin Crossb916a382016-07-29 17:28:03 -0700453 *baseCompiler
454 *baseLinker
455 *baseInstaller
Inseob Kimeda2e9c2020-03-03 22:06:32 +0900456
sophiez4c4f8032021-08-16 22:54:00 -0700457 apiListCoverageXmlPath android.ModuleOutPath
Spandan Dase20c56c2024-07-23 21:34:24 +0000458
459 // Path to the file containing the APIs exported by this library
460 stubsSymbolFilePath android.Path
Ivan Lozano2afdfe62025-02-27 04:19:23 -0800461
462 // Forces production of the generated Rust staticlib for cc_library_static.
463 // Intended to be used to provide these generated staticlibs for Make.
464 wideStaticlibForMake bool
Inseob Kimeda2e9c2020-03-03 22:06:32 +0900465}
466
Chris Parsons3c27ca32020-11-20 12:42:07 -0500467// linkerProps returns the list of properties structs relevant for this library. (For example, if
468// the library is cc_shared_library, then static-library properties are omitted.)
Colin Crossb916a382016-07-29 17:28:03 -0700469func (library *libraryDecorator) linkerProps() []interface{} {
470 var props []interface{}
471 props = append(props, library.baseLinker.linkerProps()...)
Colin Crosse1bb5d02019-09-24 14:55:04 -0700472 props = append(props,
Colin Cross4d9c2d12016-07-29 12:48:20 -0700473 &library.Properties,
Colin Crossa48ab5b2017-02-14 15:28:44 -0800474 &library.MutatedProperties,
Colin Cross4d9c2d12016-07-29 12:48:20 -0700475 &library.flagExporter.Properties,
Colin Cross22f37952018-09-05 10:43:13 -0700476 &library.stripper.StripProperties)
Colin Crosse1bb5d02019-09-24 14:55:04 -0700477
478 if library.MutatedProperties.BuildShared {
479 props = append(props, &library.SharedProperties)
480 }
481 if library.MutatedProperties.BuildStatic {
482 props = append(props, &library.StaticProperties)
483 }
484
485 return props
Colin Cross4d9c2d12016-07-29 12:48:20 -0700486}
487
Ivan Lozanoa8a1fa12024-10-30 18:15:59 +0000488func CommonLibraryLinkerFlags(ctx android.ModuleContext, flags Flags,
Ivan Lozano18ac30a2024-10-30 18:04:06 +0000489 toolchain config.Toolchain, libName string) Flags {
490
491 mod, ok := ctx.Module().(LinkableInterface)
492
493 if !ok {
494 ctx.ModuleErrorf("trying to add linker flags to a non-LinkableInterface module.")
495 return flags
496 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700497
Colin Crossb916a382016-07-29 17:28:03 -0700498 // MinGW spits out warnings about -fPIC even for -fpie?!) being ignored because
499 // all code is position independent, and then those warnings get promoted to
500 // errors.
Colin Cross3edeee12017-04-04 12:59:48 -0700501 if !ctx.Windows() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800502 flags.Global.CFlags = append(flags.Global.CFlags, "-fPIC")
Colin Crossb916a382016-07-29 17:28:03 -0700503 }
Ivan Lozano18ac30a2024-10-30 18:04:06 +0000504 if mod.Shared() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700505 var f []string
Ivan Lozano18ac30a2024-10-30 18:04:06 +0000506 if toolchain.Bionic() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700507 f = append(f,
508 "-nostdlib",
509 "-Wl,--gc-sections",
510 )
511 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700512 if ctx.Darwin() {
513 f = append(f,
514 "-dynamiclib",
Ivan Lozano18ac30a2024-10-30 18:04:06 +0000515 "-install_name @rpath/"+libName+toolchain.ShlibSuffix(),
Colin Cross4d9c2d12016-07-29 12:48:20 -0700516 )
Colin Cross7863cf52016-10-20 10:47:21 -0700517 if ctx.Arch().ArchType == android.X86 {
518 f = append(f,
519 "-read_only_relocs suppress",
520 )
521 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700522 } else {
Josh Gao75a50a22019-06-07 17:58:59 -0700523 f = append(f, "-shared")
524 if !ctx.Windows() {
Ivan Lozano18ac30a2024-10-30 18:04:06 +0000525 f = append(f, "-Wl,-soname,"+libName+toolchain.ShlibSuffix())
Josh Gao75a50a22019-06-07 17:58:59 -0700526 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700527 }
Colin Cross4af21ed2019-11-04 09:37:55 -0800528 flags.Global.LdFlags = append(flags.Global.LdFlags, f...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700529 }
530
531 return flags
532}
533
Ivan Lozano18ac30a2024-10-30 18:04:06 +0000534// linkerFlags takes a Flags struct and augments it to contain linker flags that are defined by this
535// library, or that are implied by attributes of this library (such as whether this library is a
536// shared library).
537func (library *libraryDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
538 flags = library.baseLinker.linkerFlags(ctx, flags)
Ivan Lozanoa8a1fa12024-10-30 18:15:59 +0000539 flags = CommonLibraryLinkerFlags(ctx, flags, ctx.toolchain(), library.getLibName(ctx))
Ivan Lozano18ac30a2024-10-30 18:04:06 +0000540 if library.static() {
541 flags.Local.CFlags = append(flags.Local.CFlags, library.StaticProperties.Static.Cflags.GetOrDefault(ctx, nil)...)
542 } else if library.shared() {
543 flags.Local.CFlags = append(flags.Local.CFlags, library.SharedProperties.Shared.Cflags.GetOrDefault(ctx, nil)...)
544 }
545
546 return flags
547}
548
Chris Parsons3c27ca32020-11-20 12:42:07 -0500549// compilerFlags takes a Flags and augments it to contain compile flags from global values,
550// per-target values, module type values, per-module Blueprints properties, extra flags from
551// `flags`, and generated sources from `deps`.
Colin Crossf18e1102017-11-16 14:33:08 -0800552func (library *libraryDecorator) compilerFlags(ctx ModuleContext, flags Flags, deps PathDeps) Flags {
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700553 exportIncludeDirs := library.flagExporter.exportedIncludes(ctx)
Dan Willemsen273af7f2016-11-03 15:53:42 -0700554 if len(exportIncludeDirs) > 0 {
Colin Crossdad8c952017-04-26 14:55:27 -0700555 f := includeDirsToFlags(exportIncludeDirs)
Colin Cross4af21ed2019-11-04 09:37:55 -0800556 flags.Local.CommonFlags = append(flags.Local.CommonFlags, f)
557 flags.Local.YasmFlags = append(flags.Local.YasmFlags, f)
Dan Willemsen273af7f2016-11-03 15:53:42 -0700558 }
559
Jiyong Park7ed9de32018-10-15 22:25:07 +0900560 flags = library.baseCompiler.compilerFlags(ctx, flags, deps)
Colin Cross127bb8b2020-12-16 16:46:01 -0800561 if ctx.IsLlndk() {
562 // LLNDK libraries ignore most of the properties on the cc_library and use the
563 // LLNDK-specific properties instead.
564 // Wipe all the module-local properties, leaving only the global properties.
565 flags.Local = LocalOrGlobalFlags{}
566 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000567 if library.BuildStubs() {
Jiyong Park64379952018-12-13 18:37:29 +0900568 // Remove -include <file> when compiling stubs. Otherwise, the force included
569 // headers might cause conflicting types error with the symbols in the
570 // generated stubs source code. e.g.
571 // double acos(double); // in header
572 // void acos() {} // in the generated source code
573 removeInclude := func(flags []string) []string {
574 ret := flags[:0]
575 for _, f := range flags {
576 if strings.HasPrefix(f, "-include ") {
577 continue
578 }
579 ret = append(ret, f)
580 }
581 return ret
582 }
Colin Cross4af21ed2019-11-04 09:37:55 -0800583 flags.Local.CommonFlags = removeInclude(flags.Local.CommonFlags)
584 flags.Local.CFlags = removeInclude(flags.Local.CFlags)
Jiyong Park64379952018-12-13 18:37:29 +0900585
Ivan Lozanoa8a1fa12024-10-30 18:15:59 +0000586 flags = AddStubLibraryCompilerFlags(flags)
Jiyong Park7ed9de32018-10-15 22:25:07 +0900587 }
588 return flags
Dan Willemsen273af7f2016-11-03 15:53:42 -0700589}
590
Colin Cross91ae5ec2024-10-01 14:03:40 -0700591func (library *libraryDecorator) getHeaderAbiCheckerProperties(m *Module) headerAbiCheckerProperties {
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +0800592 variantProps := &library.Properties.Target.Platform.Header_abi_checker
593 if m.InVendor() {
594 variantProps = &library.Properties.Target.Vendor.Header_abi_checker
595 } else if m.InProduct() {
596 variantProps = &library.Properties.Target.Product.Header_abi_checker
597 }
598 props := library.Properties.Header_abi_checker
599 err := proptools.AppendProperties(&props, variantProps, nil)
600 if err != nil {
Colin Cross91ae5ec2024-10-01 14:03:40 -0700601 panic(fmt.Errorf("Cannot merge headerAbiCheckerProperties: %s", err.Error()))
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +0800602 }
603 return props
Logan Chien41eabe62019-04-10 13:33:58 +0800604}
605
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700606func (library *libraryDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000607 sharedFlags := ctx.getSharedFlags()
608
Colin Cross127bb8b2020-12-16 16:46:01 -0800609 if ctx.IsLlndk() {
Justin Yun22abf212024-10-10 16:22:09 +0900610 // Get the matching SDK version for the vendor API level.
611 version, err := android.GetSdkVersionForVendorApiLevel(ctx.Config().VendorApiLevel())
612 if err != nil {
613 panic(err)
Jooyung Han33eb6152024-03-11 15:46:48 +0900614 }
Justin Yun22abf212024-10-10 16:22:09 +0900615
Justin Yune16abc62025-03-06 16:24:22 +0900616 llndkFlag := "--llndk"
617 if ctx.baseModuleName() == "libbinder_ndk" && ctx.inProduct() {
618 // This is a special case only for the libbinder_ndk. As the product partition is in the
619 // framework side along with system and system_ext partitions in Treble, libbinder_ndk
620 // provides different binder interfaces between product and vendor modules.
621 // In libbinder_ndk, 'llndk' annotation is for the vendor APIs; while 'systemapi'
622 // annotation is for the product APIs.
623 // Use '--systemapi' flag for building the llndk stub of product variant for the
624 // libbinder_ndk.
625 llndkFlag = "--systemapi"
626 }
627
Colin Cross127bb8b2020-12-16 16:46:01 -0800628 // This is the vendor variant of an LLNDK library, build the LLNDK stubs.
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000629 nativeAbiResult := ParseNativeAbiDefinition(ctx,
Dan Albertf1d14c72020-07-30 14:32:55 -0700630 String(library.Properties.Llndk.Symbol_file),
Justin Yune16abc62025-03-06 16:24:22 +0900631 nativeClampedApiLevel(ctx, version), llndkFlag)
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000632 objs := CompileStubLibrary(ctx, flags, nativeAbiResult.StubSrc, sharedFlags)
Colin Cross127bb8b2020-12-16 16:46:01 -0800633 if !Bool(library.Properties.Llndk.Unversioned) {
Dan Albertf1d14c72020-07-30 14:32:55 -0700634 library.versionScriptPath = android.OptionalPathForPath(
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000635 nativeAbiResult.VersionScript)
Colin Cross127bb8b2020-12-16 16:46:01 -0800636 }
637 return objs
638 }
Colin Cross5271fea2021-04-27 13:06:04 -0700639 if ctx.IsVendorPublicLibrary() {
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000640 nativeAbiResult := ParseNativeAbiDefinition(ctx,
Dan Albertf1d14c72020-07-30 14:32:55 -0700641 String(library.Properties.Vendor_public_library.Symbol_file),
642 android.FutureApiLevel, "")
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000643 objs := CompileStubLibrary(ctx, flags, nativeAbiResult.StubSrc, sharedFlags)
Colin Cross5271fea2021-04-27 13:06:04 -0700644 if !Bool(library.Properties.Vendor_public_library.Unversioned) {
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000645 library.versionScriptPath = android.OptionalPathForPath(nativeAbiResult.VersionScript)
Colin Cross5271fea2021-04-27 13:06:04 -0700646 }
647 return objs
648 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000649 if library.BuildStubs() {
Spandan Das357ffcc2024-07-24 18:07:48 +0000650 return library.compileModuleLibApiStubs(ctx, flags, deps)
Jiyong Park7ed9de32018-10-15 22:25:07 +0900651 }
652
Cole Faust96a692b2024-08-08 14:47:51 -0700653 srcs := library.baseCompiler.Properties.Srcs.GetOrDefault(ctx, nil)
654 staticSrcs := library.StaticProperties.Static.Srcs.GetOrDefault(ctx, nil)
655 sharedSrcs := library.SharedProperties.Shared.Srcs.GetOrDefault(ctx, nil)
Colin Cross5950f382016-12-13 12:50:57 -0800656 if !library.buildShared() && !library.buildStatic() {
Cole Faust96a692b2024-08-08 14:47:51 -0700657 if len(srcs) > 0 {
Colin Cross5950f382016-12-13 12:50:57 -0800658 ctx.PropertyErrorf("srcs", "cc_library_headers must not have any srcs")
659 }
Cole Faust96a692b2024-08-08 14:47:51 -0700660 if len(staticSrcs) > 0 {
Colin Cross5950f382016-12-13 12:50:57 -0800661 ctx.PropertyErrorf("static.srcs", "cc_library_headers must not have any srcs")
662 }
Cole Faust96a692b2024-08-08 14:47:51 -0700663 if len(sharedSrcs) > 0 {
Colin Cross5950f382016-12-13 12:50:57 -0800664 ctx.PropertyErrorf("shared.srcs", "cc_library_headers must not have any srcs")
665 }
666 return Objects{}
667 }
Yo Chiang2bbadfa2020-12-14 11:42:16 +0800668 if library.sabi.shouldCreateSourceAbiDump() {
Hsin-Yi Chenaf369882024-03-29 20:10:36 +0800669 dirs := library.exportedIncludeDirsForAbiCheck(ctx)
Hsin-Yi Chenea748242025-01-15 15:44:48 +0800670 flags.SAbiFlags = make([]string, 0, len(dirs)+1)
Hsin-Yi Chenaf369882024-03-29 20:10:36 +0800671 for _, dir := range dirs {
672 flags.SAbiFlags = append(flags.SAbiFlags, "-I"+dir)
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -0800673 }
Hsin-Yi Chenea748242025-01-15 15:44:48 +0800674 // If this library does not export any include directory, do not append the flags
675 // so that the ABI tool dumps everything without filtering by the include directories.
676 // requiresGlobalIncludes returns whether this library can include CommonGlobalIncludes.
677 // If the library cannot include them, it cannot export them.
678 if len(dirs) > 0 && requiresGlobalIncludes(ctx) {
679 flags.SAbiFlags = append(flags.SAbiFlags, "${config.CommonGlobalIncludes}")
680 }
Cole Faust96a692b2024-08-08 14:47:51 -0700681 totalLength := len(srcs) + len(deps.GeneratedSources) +
682 len(sharedSrcs) + len(staticSrcs)
Jaewoong Jung18aefc12020-12-21 09:11:10 -0800683 if totalLength > 0 {
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -0800684 flags.SAbiDump = true
685 }
686 }
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700687 objs := library.baseCompiler.compile(ctx, flags, deps)
688 library.reuseObjects = objs
Colin Cross2f336352016-10-26 10:03:47 -0700689 buildFlags := flagsToBuilderFlags(flags)
Colin Crossb916a382016-07-29 17:28:03 -0700690
Colin Cross4d9c2d12016-07-29 12:48:20 -0700691 if library.static() {
Cole Faust96a692b2024-08-08 14:47:51 -0700692 srcs := android.PathsForModuleSrc(ctx, staticSrcs)
Chih-Hung Hsieh769a51c2021-09-17 17:18:39 -0700693 objs = objs.Append(compileObjs(ctx, buildFlags, android.DeviceStaticLibrary, srcs,
694 android.PathsForModuleSrc(ctx, library.StaticProperties.Static.Tidy_disabled_srcs),
Chih-Hung Hsieh9db8a0c2022-02-17 12:54:45 -0800695 android.PathsForModuleSrc(ctx, library.StaticProperties.Static.Tidy_timeout_srcs),
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000696 library.baseCompiler.pathDeps, library.baseCompiler.cFlagsDeps, sharedFlags))
Colin Crossa48ab5b2017-02-14 15:28:44 -0800697 } else if library.shared() {
Cole Faust96a692b2024-08-08 14:47:51 -0700698 srcs := android.PathsForModuleSrc(ctx, sharedSrcs)
Chih-Hung Hsieh769a51c2021-09-17 17:18:39 -0700699 objs = objs.Append(compileObjs(ctx, buildFlags, android.DeviceSharedLibrary, srcs,
700 android.PathsForModuleSrc(ctx, library.SharedProperties.Shared.Tidy_disabled_srcs),
Chih-Hung Hsieh9db8a0c2022-02-17 12:54:45 -0800701 android.PathsForModuleSrc(ctx, library.SharedProperties.Shared.Tidy_timeout_srcs),
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000702 library.baseCompiler.pathDeps, library.baseCompiler.cFlagsDeps, sharedFlags))
Colin Crossb916a382016-07-29 17:28:03 -0700703 }
704
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700705 return objs
Colin Crossb916a382016-07-29 17:28:03 -0700706}
707
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000708type ApiStubsParams struct {
709 NotInPlatform bool
710 IsNdk bool
711 BaseModuleName string
712 ModuleName string
713}
714
715// GetApiStubsFlags calculates the genstubFlags string to pass to ParseNativeAbiDefinition
716func GetApiStubsFlags(api ApiStubsParams) string {
717 var flag string
718
719 // b/239274367 --apex and --systemapi filters symbols tagged with # apex and #
720 // systemapi, respectively. The former is for symbols defined in platform libraries
721 // and the latter is for symbols defined in APEXes.
722 // A single library can contain either # apex or # systemapi, but not both.
723 // The stub generator (ndkstubgen) is additive, so passing _both_ of these to it should be a no-op.
724 // However, having this distinction helps guard accidental
725 // promotion or demotion of API and also helps the API review process b/191371676
726 if api.NotInPlatform {
727 flag = "--apex"
728 } else {
729 flag = "--systemapi"
730 }
731
732 // b/184712170, unless the lib is an NDK library, exclude all public symbols from
733 // the stub so that it is mandated that all symbols are explicitly marked with
734 // either apex or systemapi.
735 if !api.IsNdk &&
736 // the symbol files of libclang libs are autogenerated and do not contain systemapi tags
737 // TODO (spandandas): Update mapfile.py to include #systemapi tag on all symbols
738 !strings.Contains(api.ModuleName, "libclang_rt") {
739 flag = flag + " --no-ndk"
740 }
741
742 // TODO(b/361303067): Remove this special case if bionic/ projects are added to ART development branches.
743 if isBionic(api.BaseModuleName) {
744 // set the flags explicitly for bionic libs.
745 // this is necessary for development in minimal branches which does not contain bionic/*.
746 // In such minimal branches, e.g. on the prebuilt libc stubs
747 // 1. IsNdk will return false (since the ndk_library definition for libc does not exist)
748 // 2. NotInPlatform will return true (since the source com.android.runtime does not exist)
749 flag = "--apex"
750 }
751
752 return flag
753}
754
Spandan Das357ffcc2024-07-24 18:07:48 +0000755// Compile stubs for the API surface between platform and apex
756// This method will be used by source and prebuilt cc module types.
757func (library *libraryDecorator) compileModuleLibApiStubs(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
758 // TODO (b/275273834): Make this a hard error when the symbol files have been added to module sdk.
759 if library.Properties.Stubs.Symbol_file == nil {
760 return Objects{}
761 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000762
Spandan Das357ffcc2024-07-24 18:07:48 +0000763 symbolFile := String(library.Properties.Stubs.Symbol_file)
764 library.stubsSymbolFilePath = android.PathForModuleSrc(ctx, symbolFile)
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000765
766 apiParams := ApiStubsParams{
767 NotInPlatform: ctx.notInPlatform(),
768 IsNdk: ctx.Module().(*Module).IsNdk(ctx.Config()),
769 BaseModuleName: ctx.baseModuleName(),
770 ModuleName: ctx.ModuleName(),
Spandan Das357ffcc2024-07-24 18:07:48 +0000771 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000772 flag := GetApiStubsFlags(apiParams)
773
774 nativeAbiResult := ParseNativeAbiDefinition(ctx, symbolFile,
Spandan Das357ffcc2024-07-24 18:07:48 +0000775 android.ApiLevelOrPanic(ctx, library.MutatedProperties.StubsVersion), flag)
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000776 objs := CompileStubLibrary(ctx, flags, nativeAbiResult.StubSrc, ctx.getSharedFlags())
Spandan Das357ffcc2024-07-24 18:07:48 +0000777
778 library.versionScriptPath = android.OptionalPathForPath(
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000779 nativeAbiResult.VersionScript)
Spandan Das357ffcc2024-07-24 18:07:48 +0000780 // Parse symbol file to get API list for coverage
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000781 if library.StubsVersion() == "current" && ctx.PrimaryArch() && !ctx.inRecovery() && !ctx.inProduct() && !ctx.inVendor() {
Ivan Lozanoa8a1fa12024-10-30 18:15:59 +0000782 library.apiListCoverageXmlPath = ParseSymbolFileForAPICoverage(ctx, symbolFile)
Spandan Das357ffcc2024-07-24 18:07:48 +0000783 }
784
785 return objs
786}
787
Colin Crossb916a382016-07-29 17:28:03 -0700788type libraryInterface interface {
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000789 VersionedInterface
Colin Cross3572cf72020-10-01 15:58:11 -0700790
Colin Crossb916a382016-07-29 17:28:03 -0700791 static() bool
Inseob Kimae553032019-05-14 18:52:49 +0900792 shared() bool
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700793 objs() Objects
Colin Cross0de8a1e2020-09-18 14:15:30 -0700794 reuseObjs() Objects
Colin Cross26c34ed2016-09-30 17:10:16 -0700795 toc() android.OptionalPath
Colin Crossb916a382016-07-29 17:28:03 -0700796
797 // Returns true if the build options for the module have selected a static or shared build
798 buildStatic() bool
799 buildShared() bool
800
801 // Sets whether a specific variant is static or shared
Colin Crossa48ab5b2017-02-14 15:28:44 -0800802 setStatic()
803 setShared()
Logan Chien41eabe62019-04-10 13:33:58 +0800804
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +0800805 // Gets the ABI properties for vendor, product, or platform variant
Colin Cross91ae5ec2024-10-01 14:03:40 -0700806 getHeaderAbiCheckerProperties(m *Module) headerAbiCheckerProperties
Yo Chiang2bbadfa2020-12-14 11:42:16 +0800807
Logan Chien41eabe62019-04-10 13:33:58 +0800808 // Write LOCAL_ADDITIONAL_DEPENDENCIES for ABI diff
809 androidMkWriteAdditionalDependenciesForSourceAbiDiff(w io.Writer)
Jiyong Parka90ca002019-10-07 15:47:24 +0900810
Yu Liub73c3a62024-12-10 00:58:06 +0000811 apexAvailable() []string
Colin Crossceaa5322021-09-28 16:37:50 -0700812
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000813 setAPIListCoverageXMLPath(out android.ModuleOutPath)
814 symbolsFile() *string
815 setSymbolFilePath(path android.Path)
816 setVersionScriptPath(path android.OptionalPath)
Colin Cross1bc94122021-10-28 13:25:54 -0700817
818 installable() *bool
Colin Crossb916a382016-07-29 17:28:03 -0700819}
820
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000821func (library *libraryDecorator) symbolsFile() *string {
822 return library.Properties.Stubs.Symbol_file
823}
Colin Crossc88c2722020-09-28 17:32:47 -0700824
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000825func (library *libraryDecorator) setSymbolFilePath(path android.Path) {
826 library.stubsSymbolFilePath = path
827}
Colin Cross0477b422020-10-13 18:43:54 -0700828
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000829func (library *libraryDecorator) setVersionScriptPath(path android.OptionalPath) {
830 library.versionScriptPath = path
831}
832
833type VersionedInterface interface {
834 BuildStubs() bool
835 SetBuildStubs(isLatest bool)
836 HasStubsVariants() bool
837 IsStubsImplementationRequired() bool
838 SetStubsVersion(string)
839 StubsVersion() string
840
841 StubsVersions(ctx android.BaseModuleContext) []string
842 SetAllStubsVersions([]string)
843 AllStubsVersions() []string
844
845 ImplementationModuleName(name string) string
846 HasLLNDKStubs() bool
847 HasLLNDKHeaders() bool
848 HasVendorPublicLibrary() bool
849 IsLLNDKMovedToApex() bool
850
851 GetAPIListCoverageXMLPath() android.ModuleOutPath
Colin Crossc88c2722020-09-28 17:32:47 -0700852}
853
854var _ libraryInterface = (*libraryDecorator)(nil)
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000855var _ VersionedInterface = (*libraryDecorator)(nil)
Colin Crossc88c2722020-09-28 17:32:47 -0700856
Justin Yun6977e8a2020-10-29 18:24:11 +0900857func (library *libraryDecorator) getLibNameHelper(baseModuleName string, inVendor bool, inProduct bool) string {
Colin Crossb916a382016-07-29 17:28:03 -0700858 name := library.libName
859 if name == "" {
dimitryd95964a2018-11-07 13:43:34 +0100860 name = String(library.Properties.Stem)
861 if name == "" {
Jooyung Han0302a842019-10-30 18:43:49 +0900862 name = baseModuleName
dimitryd95964a2018-11-07 13:43:34 +0100863 }
Colin Crossb916a382016-07-29 17:28:03 -0700864 }
865
Colin Cross0fd6a412019-08-16 14:22:10 -0700866 suffix := ""
Justin Yun6977e8a2020-10-29 18:24:11 +0900867 if inVendor {
Colin Cross0fd6a412019-08-16 14:22:10 -0700868 suffix = String(library.Properties.Target.Vendor.Suffix)
Justin Yun6977e8a2020-10-29 18:24:11 +0900869 } else if inProduct {
870 suffix = String(library.Properties.Target.Product.Suffix)
Colin Cross0fd6a412019-08-16 14:22:10 -0700871 }
872 if suffix == "" {
873 suffix = String(library.Properties.Suffix)
874 }
875
Jooyung Han0302a842019-10-30 18:43:49 +0900876 return name + suffix
877}
878
Chris Parsons3c27ca32020-11-20 12:42:07 -0500879// getLibName returns the actual canonical name of the library (the name which
880// should be passed to the linker via linker flags).
Jooyung Han0302a842019-10-30 18:43:49 +0900881func (library *libraryDecorator) getLibName(ctx BaseModuleContext) string {
Justin Yun6977e8a2020-10-29 18:24:11 +0900882 name := library.getLibNameHelper(ctx.baseModuleName(), ctx.inVendor(), ctx.inProduct())
Colin Cross0fd6a412019-08-16 14:22:10 -0700883
Colin Crossb916a382016-07-29 17:28:03 -0700884 if ctx.Host() && Bool(library.Properties.Unique_host_soname) {
885 if !strings.HasSuffix(name, "-host") {
886 name = name + "-host"
887 }
888 }
889
Inseob Kim0ce291e2019-07-04 14:38:27 +0900890 return name
Colin Crossb916a382016-07-29 17:28:03 -0700891}
892
Jiyong Parkda732bd2018-11-02 18:23:15 +0900893var versioningMacroNamesListMutex sync.Mutex
894
Colin Crossb916a382016-07-29 17:28:03 -0700895func (library *libraryDecorator) linkerInit(ctx BaseModuleContext) {
896 location := InstallInSystem
Dan Albert61f32122018-07-26 14:00:24 -0700897 if library.baseLinker.sanitize.inSanitizerDir() {
Vishwath Mohan1dd88392017-03-29 22:00:18 -0700898 location = InstallInSanitizerDir
Colin Crossb916a382016-07-29 17:28:03 -0700899 }
900 library.baseInstaller.location = location
Colin Crossb916a382016-07-29 17:28:03 -0700901 library.baseLinker.linkerInit(ctx)
Jiyong Park7ed9de32018-10-15 22:25:07 +0900902 // Let baseLinker know whether this variant is for stubs or not, so that
903 // it can omit things that are not required for linking stubs.
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000904 library.baseLinker.dynamicProperties.BuildStubs = library.BuildStubs()
Jiyong Parkda732bd2018-11-02 18:23:15 +0900905
Ivan Lozano9eaacc82024-10-30 14:28:17 +0000906 if library.BuildStubs() {
Jiyong Parkda732bd2018-11-02 18:23:15 +0900907 macroNames := versioningMacroNamesList(ctx.Config())
908 myName := versioningMacroName(ctx.ModuleName())
909 versioningMacroNamesListMutex.Lock()
910 defer versioningMacroNamesListMutex.Unlock()
911 if (*macroNames)[myName] == "" {
912 (*macroNames)[myName] = ctx.ModuleName()
913 } else if (*macroNames)[myName] != ctx.ModuleName() {
914 ctx.ModuleErrorf("Macro name %q for versioning conflicts with macro name from module %q ", myName, (*macroNames)[myName])
915 }
916 }
Colin Crossb916a382016-07-29 17:28:03 -0700917}
918
dimitry0345ad82018-12-05 16:28:14 +0100919func (library *libraryDecorator) compilerDeps(ctx DepsContext, deps Deps) Deps {
Colin Cross127bb8b2020-12-16 16:46:01 -0800920 if ctx.IsLlndk() {
921 // LLNDK libraries ignore most of the properties on the cc_library and use the
922 // LLNDK-specific properties instead.
923 return deps
924 }
925
dimitry0345ad82018-12-05 16:28:14 +0100926 deps = library.baseCompiler.compilerDeps(ctx, deps)
927
dimitry0345ad82018-12-05 16:28:14 +0100928 return deps
929}
930
Colin Cross37047f12016-12-13 17:06:13 -0800931func (library *libraryDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
Colin Cross127bb8b2020-12-16 16:46:01 -0800932 if ctx.IsLlndk() {
933 // LLNDK libraries ignore most of the properties on the cc_library and use the
934 // LLNDK-specific properties instead.
Colin Cross0fb7fcd2021-03-02 11:00:07 -0800935 deps.HeaderLibs = append([]string(nil), library.Properties.Llndk.Export_llndk_headers...)
936 deps.ReexportHeaderLibHeaders = append([]string(nil), library.Properties.Llndk.Export_llndk_headers...)
Colin Cross127bb8b2020-12-16 16:46:01 -0800937 return deps
938 }
Colin Cross5271fea2021-04-27 13:06:04 -0700939 if ctx.IsVendorPublicLibrary() {
940 headers := library.Properties.Vendor_public_library.Export_public_headers
941 deps.HeaderLibs = append([]string(nil), headers...)
942 deps.ReexportHeaderLibHeaders = append([]string(nil), headers...)
943 return deps
944 }
Colin Cross127bb8b2020-12-16 16:46:01 -0800945
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800946 if library.static() {
Martin Stjernholm10566a02020-03-24 01:19:52 +0000947 // Compare with nil because an empty list needs to be propagated.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700948 if library.StaticProperties.Static.System_shared_libs != nil {
949 library.baseLinker.Properties.System_shared_libs = library.StaticProperties.Static.System_shared_libs
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800950 }
951 } else if library.shared() {
Martin Stjernholm10566a02020-03-24 01:19:52 +0000952 // Compare with nil because an empty list needs to be propagated.
Colin Crosse1bb5d02019-09-24 14:55:04 -0700953 if library.SharedProperties.Shared.System_shared_libs != nil {
954 library.baseLinker.Properties.System_shared_libs = library.SharedProperties.Shared.System_shared_libs
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800955 }
956 }
957
Colin Crossb916a382016-07-29 17:28:03 -0700958 deps = library.baseLinker.linkerDeps(ctx, deps)
959
960 if library.static() {
961 deps.WholeStaticLibs = append(deps.WholeStaticLibs,
Cole Faustf0006e72024-08-19 14:39:19 -0700962 library.StaticProperties.Static.Whole_static_libs.GetOrDefault(ctx, nil)...)
963 deps.StaticLibs = append(deps.StaticLibs, library.StaticProperties.Static.Static_libs.GetOrDefault(ctx, nil)...)
964 deps.SharedLibs = append(deps.SharedLibs, library.StaticProperties.Static.Shared_libs.GetOrDefault(ctx, nil)...)
Colin Crosseefe9a32019-01-22 14:41:08 -0800965
Colin Crosse1bb5d02019-09-24 14:55:04 -0700966 deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, library.StaticProperties.Static.Export_shared_lib_headers...)
967 deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, library.StaticProperties.Static.Export_static_lib_headers...)
Colin Crossa48ab5b2017-02-14 15:28:44 -0800968 } else if library.shared() {
Alex Márquez Pérez Muñíz Díaz Puras Thaureaux01ec55e2023-01-30 22:53:04 +0000969 if library.baseLinker.Properties.crt() {
Colin Crossd1a28132021-06-21 17:34:47 -0700970 deps.CrtBegin = append(deps.CrtBegin, ctx.toolchain().CrtBeginSharedLibrary()...)
971 deps.CrtEnd = append(deps.CrtEnd, ctx.toolchain().CrtEndSharedLibrary()...)
Kalesh Singhf4ffe0a2024-01-29 13:01:51 -0800972
973 }
974 if library.baseLinker.Properties.crtPadSegment() {
975 deps.CrtEnd = append(deps.CrtEnd, ctx.toolchain().CrtPadSegmentSharedLibrary()...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700976 }
Cole Faustf0006e72024-08-19 14:39:19 -0700977 deps.WholeStaticLibs = append(deps.WholeStaticLibs, library.SharedProperties.Shared.Whole_static_libs.GetOrDefault(ctx, nil)...)
978 deps.StaticLibs = append(deps.StaticLibs, library.SharedProperties.Shared.Static_libs.GetOrDefault(ctx, nil)...)
979 deps.SharedLibs = append(deps.SharedLibs, library.SharedProperties.Shared.Shared_libs.GetOrDefault(ctx, nil)...)
Colin Crosseefe9a32019-01-22 14:41:08 -0800980
Colin Crosse1bb5d02019-09-24 14:55:04 -0700981 deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, library.SharedProperties.Shared.Export_shared_lib_headers...)
982 deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, library.SharedProperties.Shared.Export_static_lib_headers...)
Hsin-Yi Chen715142a2024-03-27 16:31:16 +0800983
984 deps.LlndkHeaderLibs = append(deps.LlndkHeaderLibs, library.Properties.Llndk.Export_llndk_headers...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700985 }
Justin Yun6977e8a2020-10-29 18:24:11 +0900986 if ctx.inVendor() {
Jiyong Park52d25bd2017-10-13 09:17:01 +0900987 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
988 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Vendor.Exclude_shared_libs)
989 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
Victor Chang51271c12019-01-30 16:02:22 +0000990 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Vendor.Exclude_shared_libs)
991 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
Jiyong Park52d25bd2017-10-13 09:17:01 +0900992 }
Justin Yun6977e8a2020-10-29 18:24:11 +0900993 if ctx.inProduct() {
994 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
995 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Product.Exclude_shared_libs)
996 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
997 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Product.Exclude_shared_libs)
998 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
999 }
Jiyong Parkf9332f12018-02-01 00:54:12 +09001000 if ctx.inRecovery() {
1001 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
1002 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Recovery.Exclude_shared_libs)
1003 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
Victor Chang51271c12019-01-30 16:02:22 +00001004 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Recovery.Exclude_shared_libs)
1005 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
Jiyong Parkf9332f12018-02-01 00:54:12 +09001006 }
Yifan Hongcf4832c2020-01-21 17:04:13 -08001007 if ctx.inRamdisk() {
1008 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
1009 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_shared_libs)
1010 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
1011 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Ramdisk.Exclude_shared_libs)
1012 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
1013 }
Yifan Hong6da33c22020-10-27 15:01:21 -07001014 if ctx.inVendorRamdisk() {
1015 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
1016 deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
1017 deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
1018 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
1019 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
1020 }
Colin Cross2383f3b2018-02-06 14:40:13 -08001021
Colin Cross4d9c2d12016-07-29 12:48:20 -07001022 return deps
1023}
1024
Cole Fauste8a87832024-09-11 11:35:46 -07001025func (library *libraryDecorator) linkerSpecifiedDeps(ctx android.ConfigurableEvaluatorContext, module *Module, specifiedDeps specifiedDeps) specifiedDeps {
Cole Faustf0006e72024-08-19 14:39:19 -07001026 specifiedDeps = library.baseLinker.linkerSpecifiedDeps(ctx, module, specifiedDeps)
Paul Duffin13f02712020-03-06 12:30:43 +00001027 var properties StaticOrSharedProperties
1028 if library.static() {
1029 properties = library.StaticProperties.Static
1030 } else if library.shared() {
1031 properties = library.SharedProperties.Shared
1032 }
1033
Cole Faustf0006e72024-08-19 14:39:19 -07001034 eval := module.ConfigurableEvaluator(ctx)
1035 specifiedDeps.sharedLibs = append(specifiedDeps.sharedLibs, properties.Shared_libs.GetOrDefault(eval, nil)...)
Martin Stjernholm10566a02020-03-24 01:19:52 +00001036
1037 // Must distinguish nil and [] in system_shared_libs - ensure that [] in
1038 // either input list doesn't come out as nil.
1039 if specifiedDeps.systemSharedLibs == nil {
1040 specifiedDeps.systemSharedLibs = properties.System_shared_libs
1041 } else {
1042 specifiedDeps.systemSharedLibs = append(specifiedDeps.systemSharedLibs, properties.System_shared_libs...)
1043 }
Paul Duffin13f02712020-03-06 12:30:43 +00001044
1045 specifiedDeps.sharedLibs = android.FirstUniqueStrings(specifiedDeps.sharedLibs)
Martin Stjernholm10566a02020-03-24 01:19:52 +00001046 if len(specifiedDeps.systemSharedLibs) > 0 {
1047 // Skip this if systemSharedLibs is either nil or [], to ensure they are
1048 // retained.
1049 specifiedDeps.systemSharedLibs = android.FirstUniqueStrings(specifiedDeps.systemSharedLibs)
1050 }
Paul Duffin13f02712020-03-06 12:30:43 +00001051 return specifiedDeps
1052}
1053
Colin Cross4a9e6ec2023-12-18 15:29:41 -08001054func (library *libraryDecorator) moduleInfoJSON(ctx ModuleContext, moduleInfoJSON *android.ModuleInfoJSON) {
1055 if library.static() {
1056 moduleInfoJSON.Class = []string{"STATIC_LIBRARIES"}
1057 moduleInfoJSON.Uninstallable = true
1058 } else if library.shared() {
1059 moduleInfoJSON.Class = []string{"SHARED_LIBRARIES"}
1060 } else if library.header() {
1061 moduleInfoJSON.Class = []string{"HEADER_LIBRARIES"}
1062 moduleInfoJSON.Uninstallable = true
1063 }
1064
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001065 if library.BuildStubs() && library.StubsVersion() != "" {
1066 moduleInfoJSON.SubName += "." + library.StubsVersion()
Colin Cross4a9e6ec2023-12-18 15:29:41 -08001067 }
1068
1069 // If a library providing a stub is included in an APEX, the private APIs of the library
1070 // is accessible only inside the APEX. From outside of the APEX, clients can only use the
1071 // public APIs via the stub. To enforce this, the (latest version of the) stub gets the
1072 // name of the library. The impl library instead gets the `.bootstrap` suffix to so that
1073 // they can be exceptionally used directly when APEXes are not available (e.g. during the
1074 // very early stage in the boot process).
1075 if len(library.Properties.Stubs.Versions) > 0 && !ctx.Host() && ctx.notInPlatform() &&
1076 !ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() && !ctx.useVndk() && !ctx.static() {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001077 if library.BuildStubs() && library.isLatestStubVersion() {
Colin Cross4a9e6ec2023-12-18 15:29:41 -08001078 moduleInfoJSON.SubName = ""
1079 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001080 if !library.BuildStubs() {
Colin Cross4a9e6ec2023-12-18 15:29:41 -08001081 moduleInfoJSON.SubName = ".bootstrap"
1082 }
1083 }
1084
1085 library.baseLinker.moduleInfoJSON(ctx, moduleInfoJSON)
1086}
1087
Cole Faust5e1454a2025-03-11 15:55:59 -07001088func (library *libraryDecorator) testSuiteInfo(ctx ModuleContext) {
1089 // not a test
1090}
1091
Colin Crossb916a382016-07-29 17:28:03 -07001092func (library *libraryDecorator) linkStatic(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001093 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Cross4d9c2d12016-07-29 12:48:20 -07001094
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001095 library.objects = deps.WholeStaticLibObjs.Copy()
1096 library.objects = library.objects.Append(objs)
Colin Crossa2bcf2c2022-02-11 13:11:55 -08001097 library.wholeStaticLibsFromPrebuilts = android.CopyOfPaths(deps.WholeStaticLibsFromPrebuilts)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001098
Ivan Lozano2afdfe62025-02-27 04:19:23 -08001099 if library.wideStaticlibForMake {
1100 if generatedLib := GenerateRustStaticlib(ctx, deps.RustRlibDeps); generatedLib != nil {
1101 // WholeStaticLibsFromPrebuilts are .a files that get included whole into the resulting staticlib
1102 // so reuse that here for our Rust staticlibs because we don't have individual object files for
1103 // these.
1104 deps.WholeStaticLibsFromPrebuilts = append(deps.WholeStaticLibsFromPrebuilts, generatedLib)
1105 }
1106
1107 }
1108
Inseob Kim0ce291e2019-07-04 14:38:27 +09001109 fileName := ctx.ModuleName() + staticLibraryExtension
Colin Cross86803cf2018-02-15 14:12:26 -08001110 outputFile := android.PathForModuleOut(ctx, fileName)
Dan Willemsen581341d2017-02-09 16:16:31 -08001111 builderFlags := flagsToBuilderFlags(flags)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001112
Dan Willemsen569edc52018-11-19 09:33:29 -08001113 if Bool(library.baseLinker.Properties.Use_version_lib) {
1114 if ctx.Host() {
1115 versionedOutputFile := outputFile
1116 outputFile = android.PathForModuleOut(ctx, "unversioned", fileName)
1117 library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
1118 } else {
1119 versionedOutputFile := android.PathForModuleOut(ctx, "versioned", fileName)
Cole Faustd143f3e2025-02-24 16:18:18 -08001120 library.defaultDistFile = versionedOutputFile
Dan Willemsen569edc52018-11-19 09:33:29 -08001121 library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
1122 }
Colin Cross86803cf2018-02-15 14:12:26 -08001123 }
1124
Chih-Hung Hsieh7540a782022-01-08 19:56:09 -08001125 transformObjToStaticLib(ctx, library.objects.objFiles, deps.WholeStaticLibsFromPrebuilts, builderFlags, outputFile, nil, objs.tidyDepFiles)
Dan Willemsen581341d2017-02-09 16:16:31 -08001126
Chris Parsonsbf4f55f2020-11-23 17:02:44 -05001127 library.coverageOutputFile = transformCoverageFilesToZip(ctx, library.objects, ctx.ModuleName())
Colin Cross4d9c2d12016-07-29 12:48:20 -07001128
Colin Cross4d9c2d12016-07-29 12:48:20 -07001129 ctx.CheckbuildFile(outputFile)
1130
Colin Cross649d8172020-12-10 12:30:21 -08001131 if library.static() {
Colin Cross40213022023-12-13 15:19:49 -08001132 android.SetProvider(ctx, StaticLibraryInfoProvider, StaticLibraryInfo{
Colin Crossa2bcf2c2022-02-11 13:11:55 -08001133 StaticLibrary: outputFile,
1134 ReuseObjects: library.reuseObjects,
1135 Objects: library.objects,
1136 WholeStaticLibsFromPrebuilts: library.wholeStaticLibsFromPrebuilts,
Colin Cross0de8a1e2020-09-18 14:15:30 -07001137
Colin Crossa14fb6a2024-10-23 16:57:06 -07001138 TransitiveStaticLibrariesForOrdering: depset.NewBuilder[android.Path](depset.TOPOLOGICAL).
Colin Cross649d8172020-12-10 12:30:21 -08001139 Direct(outputFile).
1140 Transitive(deps.TranstiveStaticLibrariesForOrdering).
1141 Build(),
1142 })
1143 }
1144
1145 if library.header() {
Colin Cross40213022023-12-13 15:19:49 -08001146 android.SetProvider(ctx, HeaderLibraryInfoProvider, HeaderLibraryInfo{})
Colin Cross649d8172020-12-10 12:30:21 -08001147 }
Colin Cross0de8a1e2020-09-18 14:15:30 -07001148
Colin Cross4d9c2d12016-07-29 12:48:20 -07001149 return outputFile
1150}
1151
Chih-Hung Hsiehf6ca1b92021-12-05 18:02:50 -08001152func ndkSharedLibDeps(ctx ModuleContext) android.Paths {
1153 if ctx.Module().(*Module).IsSdkVariant() {
1154 // The NDK sysroot timestamp file depends on all the NDK
1155 // sysroot header and shared library files.
1156 return android.Paths{getNdkBaseTimestampFile(ctx)}
1157 }
1158 return nil
1159}
1160
Colin Crossb916a382016-07-29 17:28:03 -07001161func (library *libraryDecorator) linkShared(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001162 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Cross4d9c2d12016-07-29 12:48:20 -07001163
1164 var linkerDeps android.Paths
Pirama Arumuga Nainarada83ec2017-08-31 23:38:27 -07001165 linkerDeps = append(linkerDeps, flags.LdFlagsDeps...)
Chih-Hung Hsiehf6ca1b92021-12-05 18:02:50 -08001166 linkerDeps = append(linkerDeps, ndkSharedLibDeps(ctx)...)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001167
Jerome Gaillard9b454732024-12-10 18:07:19 +00001168 exportedSymbols := ctx.ExpandOptionalSource(library.Properties.Exported_symbols_list, "exported_symbols_list")
Colin Cross2383f3b2018-02-06 14:40:13 -08001169 unexportedSymbols := ctx.ExpandOptionalSource(library.Properties.Unexported_symbols_list, "unexported_symbols_list")
1170 forceNotWeakSymbols := ctx.ExpandOptionalSource(library.Properties.Force_symbols_not_weak_list, "force_symbols_not_weak_list")
1171 forceWeakSymbols := ctx.ExpandOptionalSource(library.Properties.Force_symbols_weak_list, "force_symbols_weak_list")
Colin Cross4d9c2d12016-07-29 12:48:20 -07001172 if !ctx.Darwin() {
Jerome Gaillard9b454732024-12-10 18:07:19 +00001173 if exportedSymbols.Valid() {
1174 ctx.PropertyErrorf("exported_symbols_list", "Only supported on Darwin")
1175 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001176 if unexportedSymbols.Valid() {
1177 ctx.PropertyErrorf("unexported_symbols_list", "Only supported on Darwin")
1178 }
1179 if forceNotWeakSymbols.Valid() {
1180 ctx.PropertyErrorf("force_symbols_not_weak_list", "Only supported on Darwin")
1181 }
1182 if forceWeakSymbols.Valid() {
1183 ctx.PropertyErrorf("force_symbols_weak_list", "Only supported on Darwin")
1184 }
1185 } else {
Jerome Gaillard9b454732024-12-10 18:07:19 +00001186 if exportedSymbols.Valid() {
1187 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-exported_symbols_list,"+exportedSymbols.String())
1188 linkerDeps = append(linkerDeps, exportedSymbols.Path())
1189 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001190 if unexportedSymbols.Valid() {
Colin Cross4af21ed2019-11-04 09:37:55 -08001191 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-unexported_symbols_list,"+unexportedSymbols.String())
Colin Cross4d9c2d12016-07-29 12:48:20 -07001192 linkerDeps = append(linkerDeps, unexportedSymbols.Path())
1193 }
1194 if forceNotWeakSymbols.Valid() {
Colin Cross4af21ed2019-11-04 09:37:55 -08001195 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-force_symbols_not_weak_list,"+forceNotWeakSymbols.String())
Colin Cross4d9c2d12016-07-29 12:48:20 -07001196 linkerDeps = append(linkerDeps, forceNotWeakSymbols.Path())
1197 }
1198 if forceWeakSymbols.Valid() {
Colin Cross4af21ed2019-11-04 09:37:55 -08001199 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-force_symbols_weak_list,"+forceWeakSymbols.String())
Colin Cross4d9c2d12016-07-29 12:48:20 -07001200 linkerDeps = append(linkerDeps, forceWeakSymbols.Path())
1201 }
1202 }
Colin Cross8e21aa52020-09-28 18:28:02 -07001203 if library.versionScriptPath.Valid() {
Jiyong Parkc1e7f482019-01-12 13:39:10 +09001204 linkerScriptFlags := "-Wl,--version-script," + library.versionScriptPath.String()
Colin Cross4af21ed2019-11-04 09:37:55 -08001205 flags.Local.LdFlags = append(flags.Local.LdFlags, linkerScriptFlags)
Colin Cross8e21aa52020-09-28 18:28:02 -07001206 linkerDeps = append(linkerDeps, library.versionScriptPath.Path())
Jiyong Parkc1e7f482019-01-12 13:39:10 +09001207 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001208
1209 fileName := library.getLibName(ctx) + flags.Toolchain.ShlibSuffix()
1210 outputFile := android.PathForModuleOut(ctx, fileName)
Colin Cross0de8a1e2020-09-18 14:15:30 -07001211 unstrippedOutputFile := outputFile
Colin Cross4d9c2d12016-07-29 12:48:20 -07001212
Josh Gao75a50a22019-06-07 17:58:59 -07001213 var implicitOutputs android.WritablePaths
1214 if ctx.Windows() {
1215 importLibraryPath := android.PathForModuleOut(ctx, pathtools.ReplaceExtension(fileName, "lib"))
1216
Colin Cross4af21ed2019-11-04 09:37:55 -08001217 flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,--out-implib="+importLibraryPath.String())
Josh Gao75a50a22019-06-07 17:58:59 -07001218 implicitOutputs = append(implicitOutputs, importLibraryPath)
1219 }
1220
Colin Cross4d9c2d12016-07-29 12:48:20 -07001221 builderFlags := flagsToBuilderFlags(flags)
1222
Dan Willemsen47450072021-10-19 20:24:49 -07001223 if ctx.Darwin() && deps.DarwinSecondArchOutput.Valid() {
1224 fatOutputFile := outputFile
1225 outputFile = android.PathForModuleOut(ctx, "pre-fat", fileName)
1226 transformDarwinUniversalBinary(ctx, fatOutputFile, outputFile, deps.DarwinSecondArchOutput.Path())
1227 }
1228
Colin Crossb496cfd2018-09-10 16:50:05 -07001229 // Optimize out relinking against shared libraries whose interface hasn't changed by
1230 // depending on a table of contents file instead of the library itself.
Colin Cross70dda7e2019-10-01 22:05:35 -07001231 tocFile := outputFile.ReplaceExtension(ctx, flags.Toolchain.ShlibSuffix()[1:]+".toc")
Colin Crossb496cfd2018-09-10 16:50:05 -07001232 library.tocFile = android.OptionalPathForPath(tocFile)
Ivan Lozano7b0781d2021-11-03 15:30:18 -04001233 TransformSharedObjectToToc(ctx, outputFile, tocFile)
Colin Cross89562dc2016-10-03 17:47:19 -07001234
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001235 stripFlags := flagsToStripFlags(flags)
Colin Crossadc81a02020-12-14 17:01:55 -08001236 needsStrip := library.stripper.NeedsStrip(ctx)
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001237 if library.BuildStubs() {
Colin Crossadc81a02020-12-14 17:01:55 -08001238 // No need to strip stubs libraries
1239 needsStrip = false
1240 }
1241 if needsStrip {
Yi Kongb5c34d72018-11-07 16:28:49 -08001242 if ctx.Darwin() {
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001243 stripFlags.StripUseGnuStrip = true
Yi Kongb5c34d72018-11-07 16:28:49 -08001244 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001245 strippedOutputFile := outputFile
1246 outputFile = android.PathForModuleOut(ctx, "unstripped", fileName)
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001247 library.stripper.StripExecutableOrSharedLib(ctx, outputFile, strippedOutputFile, stripFlags)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001248 }
Colin Crossb60190a2018-09-04 16:28:17 -07001249 library.unstrippedOutputFile = outputFile
1250
Colin Crossd7227f92019-09-05 14:26:33 -07001251 outputFile = maybeInjectBoringSSLHash(ctx, outputFile, library.Properties.Inject_bssl_hash, fileName)
Pete Bentley803e1612019-08-06 22:19:59 +01001252
Dan Willemsen569edc52018-11-19 09:33:29 -08001253 if Bool(library.baseLinker.Properties.Use_version_lib) {
1254 if ctx.Host() {
1255 versionedOutputFile := outputFile
1256 outputFile = android.PathForModuleOut(ctx, "unversioned", fileName)
1257 library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
1258 } else {
1259 versionedOutputFile := android.PathForModuleOut(ctx, "versioned", fileName)
Cole Faustd143f3e2025-02-24 16:18:18 -08001260 library.defaultDistFile = versionedOutputFile
Dan Willemsen569edc52018-11-19 09:33:29 -08001261
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001262 if library.stripper.NeedsStrip(ctx) {
Dan Willemsen569edc52018-11-19 09:33:29 -08001263 out := android.PathForModuleOut(ctx, "versioned-stripped", fileName)
Cole Faustd143f3e2025-02-24 16:18:18 -08001264 library.defaultDistFile = out
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001265 library.stripper.StripExecutableOrSharedLib(ctx, versionedOutputFile, out, stripFlags)
Dan Willemsen569edc52018-11-19 09:33:29 -08001266 }
1267
1268 library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
1269 }
Colin Cross86803cf2018-02-15 14:12:26 -08001270 }
1271
Wei Li5f5d2712023-12-11 15:40:29 -08001272 // Generate an output file for dist as if strip: "all" is set on the module.
1273 // Currently this is for layoutlib release process only.
1274 for _, dist := range ctx.Module().(*Module).Dists() {
1275 if dist.Tag != nil && *dist.Tag == "stripped_all" {
1276 strippedAllOutputFile := android.PathForModuleOut(ctx, "stripped_all", fileName)
1277 transformStrip(ctx, outputFile, strippedAllOutputFile, StripFlags{Toolchain: flags.Toolchain})
1278 library.strippedAllOutputFile = strippedAllOutputFile
1279 break
1280 }
1281 }
1282
Jiyong Park64a44f22019-01-18 14:37:08 +09001283 sharedLibs := deps.EarlySharedLibs
1284 sharedLibs = append(sharedLibs, deps.SharedLibs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001285 sharedLibs = append(sharedLibs, deps.LateSharedLibs...)
1286
Jiyong Park64a44f22019-01-18 14:37:08 +09001287 linkerDeps = append(linkerDeps, deps.EarlySharedLibsDeps...)
Colin Cross26c34ed2016-09-30 17:10:16 -07001288 linkerDeps = append(linkerDeps, deps.SharedLibsDeps...)
1289 linkerDeps = append(linkerDeps, deps.LateSharedLibsDeps...)
Ivan Lozano0a468a42024-05-13 21:03:34 -04001290
Ivan Lozano85f00cf2025-02-11 20:19:19 +00001291 if generatedLib := GenerateRustStaticlib(ctx, deps.RustRlibDeps); generatedLib != nil && !library.BuildStubs() {
Ivan Lozanofd47b1a2024-05-17 14:13:41 -04001292 if ctx.Module().(*Module).WholeRustStaticlib {
1293 deps.WholeStaticLibs = append(deps.WholeStaticLibs, generatedLib)
1294 } else {
1295 deps.StaticLibs = append(deps.StaticLibs, generatedLib)
1296 }
Ivan Lozano0a468a42024-05-13 21:03:34 -04001297 }
1298
Chris Parsonsbf4f55f2020-11-23 17:02:44 -05001299 transformObjToDynamicBinary(ctx, objs.objFiles, sharedLibs,
Ivan Lozano0a468a42024-05-13 21:03:34 -04001300 deps.StaticLibs, deps.LateStaticLibs, deps.WholeStaticLibs, linkerDeps, deps.CrtBegin,
1301 deps.CrtEnd, false, builderFlags, outputFile, implicitOutputs, objs.tidyDepFiles)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001302
Dan Willemsen581341d2017-02-09 16:16:31 -08001303 objs.coverageFiles = append(objs.coverageFiles, deps.StaticLibObjs.coverageFiles...)
1304 objs.coverageFiles = append(objs.coverageFiles, deps.WholeStaticLibObjs.coverageFiles...)
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -08001305 objs.sAbiDumpFiles = append(objs.sAbiDumpFiles, deps.StaticLibObjs.sAbiDumpFiles...)
1306 objs.sAbiDumpFiles = append(objs.sAbiDumpFiles, deps.WholeStaticLibObjs.sAbiDumpFiles...)
1307
Chris Parsonsbf4f55f2020-11-23 17:02:44 -05001308 library.coverageOutputFile = transformCoverageFilesToZip(ctx, objs, library.getLibName(ctx))
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001309 library.linkSAbiDumpFiles(ctx, deps, objs, fileName, unstrippedOutputFile)
Dan Willemsen581341d2017-02-09 16:16:31 -08001310
Colin Crossa14fb6a2024-10-23 16:57:06 -07001311 var transitiveStaticLibrariesForOrdering depset.DepSet[android.Path]
Yu Liu116610a2025-01-06 21:54:48 +00001312 if static := ctx.GetDirectDepsProxyWithTag(staticVariantTag); len(static) > 0 {
Colin Cross313aa542023-12-13 13:47:44 -08001313 s, _ := android.OtherModuleProvider(ctx, static[0], StaticLibraryInfoProvider)
Liz Kammeref6dfea2021-06-08 15:37:09 -04001314 transitiveStaticLibrariesForOrdering = s.TransitiveStaticLibrariesForOrdering
Colin Cross0de8a1e2020-09-18 14:15:30 -07001315 }
1316
Colin Cross40213022023-12-13 15:19:49 -08001317 android.SetProvider(ctx, SharedLibraryInfoProvider, SharedLibraryInfo{
Liz Kammeref6dfea2021-06-08 15:37:09 -04001318 TableOfContents: android.OptionalPathForPath(tocFile),
1319 SharedLibrary: unstrippedOutputFile,
1320 TransitiveStaticLibrariesForOrdering: transitiveStaticLibrariesForOrdering,
1321 Target: ctx.Target(),
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001322 IsStubs: library.BuildStubs(),
Colin Cross0de8a1e2020-09-18 14:15:30 -07001323 })
1324
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001325 AddStubDependencyProviders(ctx)
Sam Delmerico75dbca22023-04-20 13:13:25 +00001326
1327 return unstrippedOutputFile
1328}
1329
Spandan Das357ffcc2024-07-24 18:07:48 +00001330// Visits the stub variants of the library and returns a struct containing the stub .so paths
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001331func AddStubDependencyProviders(ctx android.BaseModuleContext) []SharedStubLibrary {
Spandan Das357ffcc2024-07-24 18:07:48 +00001332 stubsInfo := []SharedStubLibrary{}
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001333 stubs := ctx.GetDirectDepsProxyWithTag(StubImplDepTag)
Colin Cross0de8a1e2020-09-18 14:15:30 -07001334 if len(stubs) > 0 {
Colin Cross0de8a1e2020-09-18 14:15:30 -07001335 for _, stub := range stubs {
Spandan Das357ffcc2024-07-24 18:07:48 +00001336 stubInfo, ok := android.OtherModuleProvider(ctx, stub, SharedLibraryInfoProvider)
1337 // TODO (b/275273834): Make this a hard error when the symbol files have been added to module sdk.
1338 if !ok {
1339 continue
1340 }
Colin Cross313aa542023-12-13 13:47:44 -08001341 flagInfo, _ := android.OtherModuleProvider(ctx, stub, FlagExporterInfoProvider)
Yu Liu8024b922024-12-20 23:31:32 +00001342 if _, ok = android.OtherModuleProvider(ctx, stub, CcInfoProvider); !ok {
1343 panic(fmt.Errorf("stub is not a cc module %s", stub))
Yu Liuffe86322024-12-18 18:53:12 +00001344 }
Chris Parsons3c27ca32020-11-20 12:42:07 -05001345 stubsInfo = append(stubsInfo, SharedStubLibrary{
Yu Liu8024b922024-12-20 23:31:32 +00001346 Version: android.OtherModuleProviderOrDefault(ctx, stub, LinkableInfoProvider).StubsVersion,
Colin Cross0de8a1e2020-09-18 14:15:30 -07001347 SharedLibraryInfo: stubInfo,
1348 FlagExporterInfo: flagInfo,
1349 })
1350 }
Spandan Das357ffcc2024-07-24 18:07:48 +00001351 if len(stubsInfo) > 0 {
1352 android.SetProvider(ctx, SharedLibraryStubsProvider, SharedLibraryStubsInfo{
1353 SharedStubLibraries: stubsInfo,
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001354 IsLLNDK: ctx.Module().(LinkableInterface).IsLlndk(),
Spandan Das357ffcc2024-07-24 18:07:48 +00001355 })
1356 }
Colin Cross0de8a1e2020-09-18 14:15:30 -07001357 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001358
Spandan Das357ffcc2024-07-24 18:07:48 +00001359 return stubsInfo
Colin Cross4d9c2d12016-07-29 12:48:20 -07001360}
1361
Jiyong Parkaf6d8952019-01-31 12:21:23 +09001362func (library *libraryDecorator) unstrippedOutputFilePath() android.Path {
1363 return library.unstrippedOutputFile
1364}
1365
Wei Li5f5d2712023-12-11 15:40:29 -08001366func (library *libraryDecorator) strippedAllOutputFilePath() android.Path {
1367 return library.strippedAllOutputFile
1368}
1369
Thiébaud Weksteend4587452020-08-19 14:53:01 +02001370func (library *libraryDecorator) disableStripping() {
1371 library.stripper.StripProperties.Strip.None = BoolPtr(true)
1372}
1373
Pirama Arumuga Nainar65c95ff2019-03-25 10:21:31 -07001374func (library *libraryDecorator) nativeCoverage() bool {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001375 if library.header() || library.BuildStubs() {
Pirama Arumuga Nainar65c95ff2019-03-25 10:21:31 -07001376 return false
1377 }
1378 return true
1379}
1380
Jiyong Parkee9a98d2019-08-09 14:44:36 +09001381func (library *libraryDecorator) coverageOutputFilePath() android.OptionalPath {
1382 return library.coverageOutputFile
1383}
1384
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001385func (library *libraryDecorator) exportedIncludeDirsForAbiCheck(ctx ModuleContext) []string {
1386 exportIncludeDirs := library.flagExporter.exportedIncludes(ctx).Strings()
1387 exportIncludeDirs = append(exportIncludeDirs, library.sabi.Properties.ReexportedIncludes...)
Hsin-Yi Chen5f228b02024-04-02 12:38:47 +08001388 exportSystemIncludeDirs := library.flagExporter.exportedSystemIncludes(ctx).Strings()
1389 exportSystemIncludeDirs = append(exportSystemIncludeDirs, library.sabi.Properties.ReexportedSystemIncludes...)
1390 // The ABI checker does not distinguish normal and system headers.
1391 return append(exportIncludeDirs, exportSystemIncludeDirs...)
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001392}
1393
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001394func (library *libraryDecorator) llndkIncludeDirsForAbiCheck(ctx ModuleContext, deps PathDeps) []string {
Hsin-Yi Chen5f228b02024-04-02 12:38:47 +08001395 var includeDirs, systemIncludeDirs []string
1396
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001397 if library.Properties.Llndk.Override_export_include_dirs != nil {
1398 includeDirs = append(includeDirs, android.PathsForModuleSrc(
1399 ctx, library.Properties.Llndk.Override_export_include_dirs).Strings()...)
1400 } else {
1401 includeDirs = append(includeDirs, library.flagExporter.exportedIncludes(ctx).Strings()...)
1402 // Ignore library.sabi.Properties.ReexportedIncludes because
1403 // LLNDK does not reexport the implementation's dependencies, such as export_header_libs.
1404 }
1405
Hsin-Yi Chen5f228b02024-04-02 12:38:47 +08001406 systemIncludeDirs = append(systemIncludeDirs,
1407 library.flagExporter.exportedSystemIncludes(ctx).Strings()...)
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001408 if Bool(library.Properties.Llndk.Export_headers_as_system) {
1409 systemIncludeDirs = append(systemIncludeDirs, includeDirs...)
1410 includeDirs = nil
1411 }
1412 // Header libs.
1413 includeDirs = append(includeDirs, deps.LlndkIncludeDirs.Strings()...)
1414 systemIncludeDirs = append(systemIncludeDirs, deps.LlndkSystemIncludeDirs.Strings()...)
1415 // The ABI checker does not distinguish normal and system headers.
1416 return append(includeDirs, systemIncludeDirs...)
1417}
1418
1419func (library *libraryDecorator) linkLlndkSAbiDumpFiles(ctx ModuleContext,
1420 deps PathDeps, sAbiDumpFiles android.Paths, soFile android.Path, libFileName string,
Hsin-Yi Chenf6fc5252024-04-15 15:43:21 +08001421 excludeSymbolVersions, excludeSymbolTags []string,
Hsin-Yi Chend6aaadc2024-11-13 16:58:46 +08001422 sdkVersionForVendorApiLevel string) android.Path {
Hsin-Yi Chenea748242025-01-15 15:44:48 +08001423 // Though LLNDK is implemented in system, the callers in vendor cannot include CommonGlobalIncludes,
1424 // so commonGlobalIncludes is false.
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001425 return transformDumpToLinkedDump(ctx,
1426 sAbiDumpFiles, soFile, libFileName+".llndk",
1427 library.llndkIncludeDirsForAbiCheck(ctx, deps),
1428 android.OptionalPathForModuleSrc(ctx, library.Properties.Llndk.Symbol_file),
1429 append([]string{"*_PLATFORM", "*_PRIVATE"}, excludeSymbolVersions...),
1430 append([]string{"platform-only"}, excludeSymbolTags...),
Hsin-Yi Chenea748242025-01-15 15:44:48 +08001431 []string{"llndk"}, sdkVersionForVendorApiLevel, false /* commonGlobalIncludes */)
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001432}
1433
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001434func (library *libraryDecorator) linkApexSAbiDumpFiles(ctx ModuleContext,
1435 deps PathDeps, sAbiDumpFiles android.Paths, soFile android.Path, libFileName string,
Hsin-Yi Chenf6fc5252024-04-15 15:43:21 +08001436 excludeSymbolVersions, excludeSymbolTags []string,
1437 sdkVersion string) android.Path {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001438 return transformDumpToLinkedDump(ctx,
1439 sAbiDumpFiles, soFile, libFileName+".apex",
1440 library.exportedIncludeDirsForAbiCheck(ctx),
1441 android.OptionalPathForModuleSrc(ctx, library.Properties.Stubs.Symbol_file),
1442 append([]string{"*_PLATFORM", "*_PRIVATE"}, excludeSymbolVersions...),
1443 append([]string{"platform-only"}, excludeSymbolTags...),
Hsin-Yi Chenea748242025-01-15 15:44:48 +08001444 []string{"apex", "systemapi"}, sdkVersion, requiresGlobalIncludes(ctx))
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001445}
1446
Hsin-Yi Chen0af4e672022-11-23 14:39:46 +08001447func getRefAbiDumpFile(ctx android.ModuleInstallPathContext,
1448 versionedDumpDir, fileName string) android.OptionalPath {
Mu-Le Leef5ed30b2022-08-30 10:37:21 +00001449
1450 currentArchType := ctx.Arch().ArchType
1451 primaryArchType := ctx.Config().DevicePrimaryArchType()
1452 archName := currentArchType.String()
1453 if currentArchType != primaryArchType {
1454 archName += "_" + primaryArchType.String()
1455 }
1456
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001457 return android.ExistentPathForSource(ctx, versionedDumpDir, archName, "source-based",
Hsin-Yi Chen0af4e672022-11-23 14:39:46 +08001458 fileName+".lsdump")
Mu-Le Leef5ed30b2022-08-30 10:37:21 +00001459}
1460
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001461// Return the previous and current SDK versions for cross-version ABI diff.
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001462func crossVersionAbiDiffSdkVersions(ctx ModuleContext, dumpDir string) (int, int) {
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001463 sdkVersionInt := ctx.Config().PlatformSdkVersion().FinalInt()
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001464
1465 if ctx.Config().PlatformSdkFinal() {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001466 return sdkVersionInt - 1, sdkVersionInt
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001467 } else {
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001468 // The platform SDK version can be upgraded before finalization while the corresponding abi dumps hasn't
1469 // been generated. Thus the Cross-Version Check chooses PLATFORM_SDK_VERION - 1 as previous version.
1470 // This situation could be identified by checking the existence of the PLATFORM_SDK_VERION dump directory.
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001471 versionedDumpDir := android.ExistentPathForSource(ctx,
1472 dumpDir, ctx.Config().PlatformSdkVersion().String())
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001473 if versionedDumpDir.Valid() {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001474 return sdkVersionInt, sdkVersionInt + 1
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001475 } else {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001476 return sdkVersionInt - 1, sdkVersionInt
Mu-Le Lee0a9005e2022-07-05 09:49:50 +00001477 }
1478 }
1479}
1480
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001481// Return the SDK version for same-version ABI diff.
1482func currRefAbiDumpSdkVersion(ctx ModuleContext) string {
Hsin-Yi Chen27bafd02024-01-08 18:38:42 +08001483 if ctx.Config().PlatformSdkFinal() {
Hsin-Yi Chena90bd382022-11-10 16:56:27 +08001484 // After sdk finalization, the ABI of the latest API level must be consistent with the source code,
1485 // so choose PLATFORM_SDK_VERSION as the current version.
1486 return ctx.Config().PlatformSdkVersion().String()
1487 } else {
1488 return "current"
1489 }
1490}
1491
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001492// sourceAbiDiff registers a build statement to compare linked sAbi dump files (.lsdump).
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001493func (library *libraryDecorator) sourceAbiDiff(ctx android.ModuleContext,
1494 sourceDump, referenceDump android.Path,
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001495 baseName, nameExt string, isLlndk, allowExtensions bool,
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001496 sourceVersion, errorMessage string) {
1497
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001498 extraFlags := []string{"-target-version", sourceVersion}
Colin Cross91ae5ec2024-10-01 14:03:40 -07001499 headerAbiChecker := library.getHeaderAbiCheckerProperties(ctx.Module().(*Module))
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +08001500 if Bool(headerAbiChecker.Check_all_apis) {
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001501 extraFlags = append(extraFlags, "-check-all-apis")
1502 } else {
1503 extraFlags = append(extraFlags,
1504 "-allow-unreferenced-changes",
1505 "-allow-unreferenced-elf-symbol-changes")
Hsin-Yi Chen32805e92025-01-02 14:48:00 +08001506 // The functions in standard libraries are not always declared in the headers.
1507 // Allow them to be added or removed without changing the symbols.
1508 if isBionic(ctx.ModuleName()) {
1509 extraFlags = append(extraFlags, "-allow-adding-removing-referenced-apis")
1510 }
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001511 }
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001512 if isLlndk {
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001513 extraFlags = append(extraFlags, "-consider-opaque-types-different")
1514 }
1515 if allowExtensions {
1516 extraFlags = append(extraFlags, "-allow-extensions")
1517 }
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +08001518 extraFlags = append(extraFlags, headerAbiChecker.Diff_flags...)
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001519
1520 library.sAbiDiff = append(
1521 library.sAbiDiff,
1522 transformAbiDumpToAbiDiff(ctx, sourceDump, referenceDump,
1523 baseName, nameExt, extraFlags, errorMessage))
1524}
1525
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001526func (library *libraryDecorator) crossVersionAbiDiff(ctx android.ModuleContext,
1527 sourceDump, referenceDump android.Path,
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001528 baseName, nameExt string, isLlndk bool, sourceVersion, prevDumpDir string) {
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001529
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001530 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 +08001531
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001532 library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, nameExt,
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001533 isLlndk, true /* allowExtensions */, sourceVersion, errorMessage)
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001534}
1535
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001536func (library *libraryDecorator) sameVersionAbiDiff(ctx android.ModuleContext,
1537 sourceDump, referenceDump android.Path,
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001538 baseName, nameExt string, isLlndk bool, lsdumpTagName string) {
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001539
1540 libName := strings.TrimSuffix(baseName, filepath.Ext(baseName))
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001541 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
1542
1543 targetRelease := ctx.Config().Getenv("TARGET_RELEASE")
1544 if targetRelease != "" {
1545 errorMessage += " --release " + targetRelease
1546 }
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001547
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001548 library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, nameExt,
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001549 isLlndk, false /* allowExtensions */, "current", errorMessage)
Hsin-Yi Chena6ddb142022-10-27 14:55:42 +08001550}
1551
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001552func (library *libraryDecorator) optInAbiDiff(ctx android.ModuleContext,
1553 sourceDump, referenceDump android.Path,
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001554 baseName, nameExt string, refDumpDir string, lsdumpTagName string) {
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001555
1556 libName := strings.TrimSuffix(baseName, filepath.Ext(baseName))
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001557 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
1558
1559 targetRelease := ctx.Config().Getenv("TARGET_RELEASE")
1560 if targetRelease != "" {
1561 errorMessage += " --release " + targetRelease
1562 }
1563
Hsin-Yi Chen8b85d812023-06-29 16:51:27 +08001564 // Most opt-in libraries do not have dumps for all default architectures.
1565 if ctx.Config().HasDeviceProduct() {
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001566 errorMessage += " --product " + ctx.Config().DeviceProduct()
Hsin-Yi Chen8b85d812023-06-29 16:51:27 +08001567 }
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001568
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001569 library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, nameExt,
Hsin-Yi Chen6fd28952024-05-02 15:37:30 +08001570 false /* isLlndk */, false /* allowExtensions */, "current", errorMessage)
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001571}
1572
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001573func (library *libraryDecorator) linkSAbiDumpFiles(ctx ModuleContext, deps PathDeps, objs Objects, fileName string, soFile android.Path) {
Yo Chiang2bbadfa2020-12-14 11:42:16 +08001574 if library.sabi.shouldCreateSourceAbiDump() {
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001575 exportedIncludeDirs := library.exportedIncludeDirsForAbiCheck(ctx)
Colin Cross91ae5ec2024-10-01 14:03:40 -07001576 headerAbiChecker := library.getHeaderAbiCheckerProperties(ctx.Module().(*Module))
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001577 currSdkVersion := currRefAbiDumpSdkVersion(ctx)
1578 currVendorVersion := ctx.Config().VendorApiLevel()
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001579
1580 // Generate source dumps.
1581 implDump := transformDumpToLinkedDump(ctx,
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001582 objs.sAbiDumpFiles, soFile, fileName,
1583 exportedIncludeDirs,
Hsin-Yi Chen843c0632023-09-25 15:26:30 +08001584 android.OptionalPathForModuleSrc(ctx, library.symbolFileForAbiCheck(ctx)),
1585 headerAbiChecker.Exclude_symbol_versions,
1586 headerAbiChecker.Exclude_symbol_tags,
Hsin-Yi Chenea748242025-01-15 15:44:48 +08001587 []string{} /* includeSymbolTags */, currSdkVersion, requiresGlobalIncludes(ctx))
Hsin-Yi Chen843c0632023-09-25 15:26:30 +08001588
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001589 var llndkDump, apexVariantDump android.Path
Colin Cross91ae5ec2024-10-01 14:03:40 -07001590 tags := classifySourceAbiDump(ctx.Module().(*Module))
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001591 optInTags := []lsdumpTag{}
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001592 for _, tag := range tags {
Hsin-Yi Chenf6fc5252024-04-15 15:43:21 +08001593 if tag == llndkLsdumpTag && currVendorVersion != "" {
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001594 if llndkDump == nil {
Hsin-Yi Chend6aaadc2024-11-13 16:58:46 +08001595 sdkVersion, err := android.GetSdkVersionForVendorApiLevel(currVendorVersion)
1596 if err != nil {
1597 ctx.ModuleErrorf("Cannot create %s llndk dump: %s", fileName, err)
1598 return
1599 }
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001600 // TODO(b/323447559): Evaluate if replacing sAbiDumpFiles with implDump is faster
1601 llndkDump = library.linkLlndkSAbiDumpFiles(ctx,
1602 deps, objs.sAbiDumpFiles, soFile, fileName,
1603 headerAbiChecker.Exclude_symbol_versions,
Hsin-Yi Chenf6fc5252024-04-15 15:43:21 +08001604 headerAbiChecker.Exclude_symbol_tags,
Hsin-Yi Chend6aaadc2024-11-13 16:58:46 +08001605 nativeClampedApiLevel(ctx, sdkVersion).String())
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001606 }
Colin Cross50bc3bc2024-10-25 09:53:41 -07001607 addLsdumpPath(ctx.Config(), string(tag)+":"+llndkDump.String())
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001608 } else if tag == apexLsdumpTag {
1609 if apexVariantDump == nil {
1610 apexVariantDump = library.linkApexSAbiDumpFiles(ctx,
1611 deps, objs.sAbiDumpFiles, soFile, fileName,
1612 headerAbiChecker.Exclude_symbol_versions,
1613 headerAbiChecker.Exclude_symbol_tags,
1614 currSdkVersion)
1615 }
Colin Cross50bc3bc2024-10-25 09:53:41 -07001616 addLsdumpPath(ctx.Config(), string(tag)+":"+apexVariantDump.String())
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001617 } else {
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001618 if tag.dirName() == "" {
1619 optInTags = append(optInTags, tag)
1620 }
Colin Cross50bc3bc2024-10-25 09:53:41 -07001621 addLsdumpPath(ctx.Config(), string(tag)+":"+implDump.String())
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001622 }
1623 }
1624
1625 // Diff source dumps and reference dumps.
1626 for _, tag := range tags {
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001627 dumpDirName := tag.dirName()
1628 if dumpDirName == "" {
1629 continue
1630 }
1631 dumpDir := filepath.Join("prebuilts", "abi-dumps", dumpDirName)
1632 isLlndk := (tag == llndkLsdumpTag)
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001633 isApex := (tag == apexLsdumpTag)
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001634 binderBitness := ctx.DeviceConfig().BinderBitness()
1635 nameExt := ""
1636 if isLlndk {
1637 nameExt = "llndk"
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001638 } else if isApex {
1639 nameExt = "apex"
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001640 }
1641 // Check against the previous version.
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001642 var prevVersion, currVersion string
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001643 sourceDump := implDump
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001644 // If this release config does not define VendorApiLevel, fall back to the old policy.
1645 if isLlndk && currVendorVersion != "" {
1646 prevVersion = ctx.Config().PrevVendorApiLevel()
1647 currVersion = currVendorVersion
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001648 // LLNDK dumps are generated by different rules after trunk stable.
1649 if android.IsTrunkStableVendorApiLevel(prevVersion) {
1650 sourceDump = llndkDump
1651 }
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001652 } else {
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001653 prevVersionInt, currVersionInt := crossVersionAbiDiffSdkVersions(ctx, dumpDir)
1654 prevVersion = strconv.Itoa(prevVersionInt)
1655 currVersion = strconv.Itoa(currVersionInt)
1656 // APEX dumps are generated by different rules after trunk stable.
1657 if isApex && prevVersionInt > 34 {
1658 sourceDump = apexVariantDump
1659 }
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001660 }
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001661 prevDumpDir := filepath.Join(dumpDir, prevVersion, binderBitness)
1662 prevDumpFile := getRefAbiDumpFile(ctx, prevDumpDir, fileName)
1663 if prevDumpFile.Valid() {
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001664 library.crossVersionAbiDiff(ctx, sourceDump, prevDumpFile.Path(),
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001665 fileName, nameExt+prevVersion, isLlndk, currVersion, prevDumpDir)
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001666 }
1667 // Check against the current version.
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001668 sourceDump = implDump
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001669 if isLlndk && currVendorVersion != "" {
1670 currVersion = currVendorVersion
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001671 if android.IsTrunkStableVendorApiLevel(currVersion) {
1672 sourceDump = llndkDump
1673 }
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001674 } else {
1675 currVersion = currSdkVersion
Hsin-Yi Chen98da0212024-04-14 19:08:17 +08001676 if isApex && (!ctx.Config().PlatformSdkFinal() ||
1677 ctx.Config().PlatformSdkVersion().FinalInt() > 34) {
1678 sourceDump = apexVariantDump
1679 }
Hsin-Yi Chen2c4a9772024-02-05 17:37:27 +08001680 }
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001681 currDumpDir := filepath.Join(dumpDir, currVersion, binderBitness)
1682 currDumpFile := getRefAbiDumpFile(ctx, currDumpDir, fileName)
1683 if currDumpFile.Valid() {
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001684 library.sameVersionAbiDiff(ctx, sourceDump, currDumpFile.Path(),
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001685 fileName, nameExt, isLlndk, string(tag))
Hsin-Yi Chen362c1882024-02-06 15:43:17 +08001686 }
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -08001687 }
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001688
1689 // Assert that a module is tagged with at most one of platformLsdumpTag, productLsdumpTag, or vendorLsdumpTag.
1690 if len(headerAbiChecker.Ref_dump_dirs) > 0 && len(optInTags) != 1 {
1691 ctx.ModuleErrorf("Expect exactly one opt-in lsdump tag when ref_dump_dirs are specified: %s", optInTags)
1692 return
1693 }
Hsin-Yi Chen3d5c6792024-04-22 11:47:57 +08001694 // Ensure that a module tagged with only platformLsdumpTag has ref_dump_dirs.
1695 // Android.bp in vendor projects should be cleaned up before this is enforced for vendorLsdumpTag and productLsdumpTag.
1696 if len(headerAbiChecker.Ref_dump_dirs) == 0 && len(tags) == 1 && tags[0] == platformLsdumpTag {
1697 ctx.ModuleErrorf("header_abi_checker is explicitly enabled, but no ref_dump_dirs are specified.")
1698 }
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001699 // Check against the opt-in reference dumps.
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +08001700 for i, optInDumpDir := range headerAbiChecker.Ref_dump_dirs {
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001701 optInDumpDirPath := android.PathForModuleSrc(ctx, optInDumpDir)
1702 // Ref_dump_dirs are not versioned.
1703 // They do not contain subdir for binder bitness because 64-bit binder has been mandatory.
1704 optInDumpFile := getRefAbiDumpFile(ctx, optInDumpDirPath.String(), fileName)
1705 if !optInDumpFile.Valid() {
1706 continue
1707 }
Hsin-Yi Chenaf369882024-03-29 20:10:36 +08001708 library.optInAbiDiff(ctx,
Hsin-Yi Chen64b2d032024-03-29 19:12:35 +08001709 implDump, optInDumpFile.Path(),
Hsin-Yi Chend5919c12024-02-19 15:15:30 +08001710 fileName, "opt"+strconv.Itoa(i), optInDumpDirPath.String(), string(optInTags[0]))
Hsin-Yi Chenf3630122022-11-17 14:29:43 +08001711 }
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -08001712 }
1713}
1714
Chris Parsons3c27ca32020-11-20 12:42:07 -05001715// link registers actions to link this library, and sets various fields
1716// on this library to reflect information that should be exported up the build
1717// tree (for example, exported flags and include paths).
Colin Crossb916a382016-07-29 17:28:03 -07001718func (library *libraryDecorator) link(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001719 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Cross4d9c2d12016-07-29 12:48:20 -07001720
Colin Cross127bb8b2020-12-16 16:46:01 -08001721 if ctx.IsLlndk() {
Colin Cross0fb7fcd2021-03-02 11:00:07 -08001722 // override the module's export_include_dirs with llndk.override_export_include_dirs
1723 // if it is set.
1724 if override := library.Properties.Llndk.Override_export_include_dirs; override != nil {
Aleks Todorovc9becde2024-06-10 12:51:53 +01001725 library.flagExporter.Properties.Export_include_dirs = proptools.NewConfigurable[[]string](
1726 nil,
1727 []proptools.ConfigurableCase[[]string]{
1728 proptools.NewConfigurableCase[[]string](nil, &override),
1729 },
1730 )
Colin Cross0fb7fcd2021-03-02 11:00:07 -08001731 }
1732
Colin Cross127bb8b2020-12-16 16:46:01 -08001733 if Bool(library.Properties.Llndk.Export_headers_as_system) {
1734 library.flagExporter.Properties.Export_system_include_dirs = append(
1735 library.flagExporter.Properties.Export_system_include_dirs,
Aleks Todorovc9becde2024-06-10 12:51:53 +01001736 library.flagExporter.Properties.Export_include_dirs.GetOrDefault(ctx, nil)...)
1737 library.flagExporter.Properties.Export_include_dirs = proptools.NewConfigurable[[]string](nil, nil)
Colin Cross127bb8b2020-12-16 16:46:01 -08001738 }
1739 }
1740
Colin Cross5271fea2021-04-27 13:06:04 -07001741 if ctx.IsVendorPublicLibrary() {
1742 // override the module's export_include_dirs with vendor_public_library.override_export_include_dirs
1743 // if it is set.
1744 if override := library.Properties.Vendor_public_library.Override_export_include_dirs; override != nil {
Aleks Todorovc9becde2024-06-10 12:51:53 +01001745 library.flagExporter.Properties.Export_include_dirs = proptools.NewConfigurable[[]string](
1746 nil,
1747 []proptools.ConfigurableCase[[]string]{
1748 proptools.NewConfigurableCase[[]string](nil, &override),
1749 },
1750 )
Colin Cross5271fea2021-04-27 13:06:04 -07001751 }
1752 }
1753
Chris Parsons3c27ca32020-11-20 12:42:07 -05001754 // Linking this library consists of linking `deps.Objs` (.o files in dependencies
1755 // of this library), together with `objs` (.o files created by compiling this
1756 // library).
Colin Crossad59e752017-11-16 14:29:11 -08001757 objs = deps.Objs.Copy().Append(objs)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001758 var out android.Path
Colin Crossa48ab5b2017-02-14 15:28:44 -08001759 if library.static() || library.header() {
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001760 out = library.linkStatic(ctx, flags, deps, objs)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001761 } else {
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001762 out = library.linkShared(ctx, flags, deps, objs)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001763 }
1764
Chris Parsons3c27ca32020-11-20 12:42:07 -05001765 // Export include paths and flags to be propagated up the tree.
Inseob Kim69378442019-06-03 19:10:47 +09001766 library.exportIncludes(ctx)
Alessandro Astone10e9d432020-10-01 12:59:27 +02001767 library.exportExtraFlags(ctx)
Inseob Kim69378442019-06-03 19:10:47 +09001768 library.reexportDirs(deps.ReexportedDirs...)
1769 library.reexportSystemDirs(deps.ReexportedSystemDirs...)
1770 library.reexportFlags(deps.ReexportedFlags...)
1771 library.reexportDeps(deps.ReexportedDeps...)
Inseob Kimd110f872019-12-06 13:15:38 +09001772 library.addExportedGeneratedHeaders(deps.ReexportedGeneratedHeaders...)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001773
Ivan Lozano0a468a42024-05-13 21:03:34 -04001774 if library.static() && len(deps.ReexportedRustRlibDeps) > 0 {
1775 library.reexportRustStaticDeps(deps.ReexportedRustRlibDeps...)
1776 }
1777
Chris Parsons3c27ca32020-11-20 12:42:07 -05001778 // Optionally export aidl headers.
Nan Zhang0007d812017-11-07 10:57:05 -08001779 if Bool(library.Properties.Aidl.Export_aidl_headers) {
Cole Faust96a692b2024-08-08 14:47:51 -07001780 if library.baseCompiler.hasAidl(ctx, deps) {
1781 if library.baseCompiler.hasSrcExt(ctx, ".aidl") {
Vinh Tran09581952023-05-16 16:03:20 -04001782 dir := android.PathForModuleGen(ctx, "aidl")
1783 library.reexportDirs(dir)
1784 }
1785 if len(deps.AidlLibraryInfos) > 0 {
1786 dir := android.PathForModuleGen(ctx, "aidl_library")
1787 library.reexportDirs(dir)
1788 }
Inseob Kimd110f872019-12-06 13:15:38 +09001789
Paul Duffin33056e82021-02-19 13:49:08 +00001790 library.reexportDeps(library.baseCompiler.aidlOrderOnlyDeps...)
1791 library.addExportedGeneratedHeaders(library.baseCompiler.aidlHeaders...)
Dan Willemsene1240db2016-11-03 14:28:51 -07001792 }
1793 }
1794
Chris Parsons3c27ca32020-11-20 12:42:07 -05001795 // Optionally export proto headers.
Nan Zhang0007d812017-11-07 10:57:05 -08001796 if Bool(library.Properties.Proto.Export_proto_headers) {
Cole Faust96a692b2024-08-08 14:47:51 -07001797 if library.baseCompiler.hasSrcExt(ctx, ".proto") {
Jiyong Park74955042019-10-22 20:19:51 +09001798 var includes android.Paths
Colin Cross19878da2019-03-28 14:45:07 -07001799 if flags.proto.CanonicalPathFromRoot {
Jiyong Park74955042019-10-22 20:19:51 +09001800 includes = append(includes, flags.proto.SubDir)
Colin Cross10d22312017-05-03 11:01:58 -07001801 }
Jiyong Park74955042019-10-22 20:19:51 +09001802 includes = append(includes, flags.proto.Dir)
Inseob Kim69378442019-06-03 19:10:47 +09001803 library.reexportDirs(includes...)
Inseob Kimd110f872019-12-06 13:15:38 +09001804
Paul Duffin33056e82021-02-19 13:49:08 +00001805 library.reexportDeps(library.baseCompiler.protoOrderOnlyDeps...)
1806 library.addExportedGeneratedHeaders(library.baseCompiler.protoHeaders...)
Colin Cross0c461f12016-10-20 16:11:43 -07001807 }
1808 }
1809
Inseob Kim07def122020-11-23 14:43:02 +09001810 // If the library is sysprop_library, expose either public or internal header selectively.
Cole Faust96a692b2024-08-08 14:47:51 -07001811 if library.baseCompiler.hasSrcExt(ctx, ".sysprop") {
Jiyong Park74955042019-10-22 20:19:51 +09001812 dir := android.PathForModuleGen(ctx, "sysprop", "include")
Inseob Kimc0907f12019-02-08 21:00:45 +09001813 if library.Properties.Sysprop.Platform != nil {
Inseob Kimc0907f12019-02-08 21:00:45 +09001814 isOwnerPlatform := Bool(library.Properties.Sysprop.Platform)
1815
Inseob Kim07def122020-11-23 14:43:02 +09001816 // If the owner is different from the user, expose public header. That is,
1817 // 1) if the user is product (as owner can only be platform / vendor)
Inseob Kim06410042021-01-18 15:23:28 +09001818 // 2) if the owner is platform and the client is vendor
1819 // We don't care Platform -> Vendor dependency as it's already forbidden.
1820 if ctx.Device() && (ctx.ProductSpecific() || (isOwnerPlatform && ctx.inVendor())) {
Jiyong Park74955042019-10-22 20:19:51 +09001821 dir = android.PathForModuleGen(ctx, "sysprop/public", "include")
Inseob Kimc0907f12019-02-08 21:00:45 +09001822 }
1823 }
1824
Paul Duffin37e0de52021-02-19 17:05:39 +00001825 // Make sure to only export headers which are within the include directory.
1826 _, headers := android.FilterPathListPredicate(library.baseCompiler.syspropHeaders, func(path android.Path) bool {
1827 _, isRel := android.MaybeRel(ctx, dir.String(), path.String())
1828 return isRel
1829 })
1830
Chris Parsons3c27ca32020-11-20 12:42:07 -05001831 // Add sysprop-related directories to the exported directories of this library.
Inseob Kim69378442019-06-03 19:10:47 +09001832 library.reexportDirs(dir)
Paul Duffin33056e82021-02-19 13:49:08 +00001833 library.reexportDeps(library.baseCompiler.syspropOrderOnlyDeps...)
Paul Duffin37e0de52021-02-19 17:05:39 +00001834 library.addExportedGeneratedHeaders(headers...)
Inseob Kim21f26902018-09-06 00:55:20 +09001835 }
1836
Chris Parsons3c27ca32020-11-20 12:42:07 -05001837 // Add stub-related flags if this library is a stub library.
Jiyong Park892a98f2020-12-14 09:20:00 +09001838 library.exportVersioningMacroIfNeeded(ctx)
Jiyong Parkda732bd2018-11-02 18:23:15 +09001839
Chris Parsons3c27ca32020-11-20 12:42:07 -05001840 // Propagate a Provider containing information about exported flags, deps, and include paths.
Colin Cross0de8a1e2020-09-18 14:15:30 -07001841 library.flagExporter.setProvider(ctx)
1842
Colin Cross4d9c2d12016-07-29 12:48:20 -07001843 return out
1844}
1845
Jiyong Park892a98f2020-12-14 09:20:00 +09001846func (library *libraryDecorator) exportVersioningMacroIfNeeded(ctx android.BaseModuleContext) {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001847 if library.BuildStubs() && library.StubsVersion() != "" && !library.skipAPIDefine {
Jiyong Park892a98f2020-12-14 09:20:00 +09001848 name := versioningMacroName(ctx.Module().(*Module).ImplementationModuleName(ctx))
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001849 apiLevel, err := android.ApiLevelFromUser(ctx, library.StubsVersion())
Jooyung Han11b0fbd2021-02-05 02:28:22 +09001850 if err != nil {
1851 ctx.ModuleErrorf("Can't export version macro: %s", err.Error())
1852 }
1853 library.reexportFlags("-D" + name + "=" + strconv.Itoa(apiLevel.FinalOrPreviewInt()))
Jiyong Park892a98f2020-12-14 09:20:00 +09001854 }
1855}
1856
Chris Parsons3c27ca32020-11-20 12:42:07 -05001857// buildStatic returns true if this library should be built as a static library.
Colin Crossb916a382016-07-29 17:28:03 -07001858func (library *libraryDecorator) buildStatic() bool {
Colin Crosse1bb5d02019-09-24 14:55:04 -07001859 return library.MutatedProperties.BuildStatic &&
1860 BoolDefault(library.StaticProperties.Static.Enabled, true)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001861}
1862
Chris Parsons3c27ca32020-11-20 12:42:07 -05001863// buildShared returns true if this library should be built as a shared library.
Colin Crossb916a382016-07-29 17:28:03 -07001864func (library *libraryDecorator) buildShared() bool {
Colin Crosse1bb5d02019-09-24 14:55:04 -07001865 return library.MutatedProperties.BuildShared &&
1866 BoolDefault(library.SharedProperties.Shared.Enabled, true)
Colin Cross4d9c2d12016-07-29 12:48:20 -07001867}
1868
Dan Willemsen5cb580f2016-09-26 17:33:01 -07001869func (library *libraryDecorator) objs() Objects {
1870 return library.objects
Colin Cross4d9c2d12016-07-29 12:48:20 -07001871}
1872
Colin Cross0de8a1e2020-09-18 14:15:30 -07001873func (library *libraryDecorator) reuseObjs() Objects {
1874 return library.reuseObjects
Colin Cross4d9c2d12016-07-29 12:48:20 -07001875}
1876
Colin Cross26c34ed2016-09-30 17:10:16 -07001877func (library *libraryDecorator) toc() android.OptionalPath {
1878 return library.tocFile
1879}
1880
Jiyong Parkf1194352019-02-25 11:05:47 +09001881func (library *libraryDecorator) installSymlinkToRuntimeApex(ctx ModuleContext, file android.Path) {
1882 dir := library.baseInstaller.installDir(ctx)
1883 dirOnDevice := android.InstallPathToOnDevicePath(ctx, dir)
Florian Mayer95cd6db2023-03-23 17:48:07 -07001884 // libc_hwasan has relative_install_dir set, which would mess up the dir.Base() logic.
1885 // hardcode here because it's the only target, if we have other targets that use this
1886 // we can generalise this.
1887 var target string
1888 if ctx.baseModuleName() == "libc_hwasan" {
1889 target = "/" + filepath.Join("apex", "com.android.runtime", "lib64", "bionic", "hwasan", file.Base())
1890 } else {
1891 base := dir.Base()
1892 target = "/" + filepath.Join("apex", "com.android.runtime", base, "bionic", file.Base())
1893 }
Jiyong Parkf1194352019-02-25 11:05:47 +09001894 ctx.InstallAbsoluteSymlink(dir, file.Base(), target)
Jaewoong Jung18aefc12020-12-21 09:11:10 -08001895 library.postInstallCmds = append(library.postInstallCmds, makeSymlinkCmd(dirOnDevice, file.Base(), target))
Jiyong Parkf1194352019-02-25 11:05:47 +09001896}
1897
Colin Crossb916a382016-07-29 17:28:03 -07001898func (library *libraryDecorator) install(ctx ModuleContext, file android.Path) {
Colin Crossc43ae772017-04-14 15:42:53 -07001899 if library.shared() {
Colin Cross386afeb2024-11-13 15:20:13 -08001900 translatedArch := ctx.Target().NativeBridge == android.NativeBridgeEnabled
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001901 if library.HasStubsVariants() && !ctx.Host() && !ctx.isSdkVariant() &&
1902 InstallToBootstrap(ctx.baseModuleName(), ctx.Config()) && !library.BuildStubs() &&
Colin Cross386afeb2024-11-13 15:20:13 -08001903 !translatedArch && !ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() {
Jiyong Parkf1194352019-02-25 11:05:47 +09001904 // Bionic libraries (e.g. libc.so) is installed to the bootstrap subdirectory.
1905 // The original path becomes a symlink to the corresponding file in the
1906 // runtime APEX.
Colin Cross386afeb2024-11-13 15:20:13 -08001907 if ctx.Device() {
1908 library.installSymlinkToRuntimeApex(ctx, file)
Jiyong Park429660f2019-01-16 22:31:11 +09001909 }
Colin Cross386afeb2024-11-13 15:20:13 -08001910 library.baseInstaller.subDir = "bootstrap"
Justin Yun8effde42017-06-23 19:24:43 +09001911 }
Przemyslaw Szczepaniake6fc5072019-07-12 14:06:23 +01001912
Colin Cross4d9c2d12016-07-29 12:48:20 -07001913 library.baseInstaller.install(ctx, file)
1914 }
Dan Albertf563d252017-10-13 00:29:00 -07001915
Dan Albert281f22b2017-12-13 15:03:47 -08001916 if Bool(library.Properties.Static_ndk_lib) && library.static() &&
Kiyoung Kimaa394802024-01-08 12:55:45 +09001917 !ctx.InVendorOrProduct() && !ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() && ctx.Device() &&
Jiyong Park7ed9de32018-10-15 22:25:07 +09001918 library.baseLinker.sanitize.isUnsanitizedVariant() &&
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001919 CtxIsForPlatform(ctx) && !ctx.isPreventInstall() {
Dan Albert4048bb02023-04-03 20:19:07 +00001920 installPath := getUnversionedLibraryInstallPath(ctx).Join(ctx, file.Base())
Dan Albertf563d252017-10-13 00:29:00 -07001921
1922 ctx.ModuleBuild(pctx, android.ModuleBuildParams{
1923 Rule: android.Cp,
1924 Description: "install " + installPath.Base(),
1925 Output: installPath,
1926 Input: file,
1927 })
1928
Colin Cross0875c522017-11-28 17:34:01 -08001929 library.ndkSysrootPath = installPath
Dan Albertf563d252017-10-13 00:29:00 -07001930 }
Colin Cross4d9c2d12016-07-29 12:48:20 -07001931}
1932
Paul Duffin0cb37b92020-03-04 14:52:46 +00001933func (library *libraryDecorator) everInstallable() bool {
1934 // Only shared and static libraries are installed. Header libraries (which are
1935 // neither static or shared) are not installed.
1936 return library.shared() || library.static()
1937}
1938
Colin Cross6a730042024-12-05 13:53:43 -08001939// static returns true if this library is for a "static" variant.
Colin Crossb916a382016-07-29 17:28:03 -07001940func (library *libraryDecorator) static() bool {
Colin Crossa48ab5b2017-02-14 15:28:44 -08001941 return library.MutatedProperties.VariantIsStatic
Colin Cross4d9c2d12016-07-29 12:48:20 -07001942}
1943
Colin Cross6a730042024-12-05 13:53:43 -08001944// staticLibrary returns true if this library is for a "static"" variant.
1945func (library *libraryDecorator) staticLibrary() bool {
1946 return library.static()
1947}
1948
1949// shared returns true if this library is for a "shared" variant.
Colin Crossa48ab5b2017-02-14 15:28:44 -08001950func (library *libraryDecorator) shared() bool {
1951 return library.MutatedProperties.VariantIsShared
1952}
1953
Chris Parsons3c27ca32020-11-20 12:42:07 -05001954// header returns true if this library is for a header-only variant.
Colin Crossa48ab5b2017-02-14 15:28:44 -08001955func (library *libraryDecorator) header() bool {
Chris Parsons3c27ca32020-11-20 12:42:07 -05001956 // Neither "static" nor "shared" implies this library is header-only.
Colin Crossa48ab5b2017-02-14 15:28:44 -08001957 return !library.static() && !library.shared()
1958}
1959
Chris Parsons3c27ca32020-11-20 12:42:07 -05001960// setStatic marks the library variant as "static".
Colin Crossa48ab5b2017-02-14 15:28:44 -08001961func (library *libraryDecorator) setStatic() {
1962 library.MutatedProperties.VariantIsStatic = true
1963 library.MutatedProperties.VariantIsShared = false
1964}
1965
Chris Parsons3c27ca32020-11-20 12:42:07 -05001966// setShared marks the library variant as "shared".
Colin Crossa48ab5b2017-02-14 15:28:44 -08001967func (library *libraryDecorator) setShared() {
1968 library.MutatedProperties.VariantIsStatic = false
1969 library.MutatedProperties.VariantIsShared = true
Colin Crossb916a382016-07-29 17:28:03 -07001970}
1971
Chris Parsons3c27ca32020-11-20 12:42:07 -05001972// BuildOnlyStatic disables building this library as a shared library.
Colin Crossab3b7322016-12-09 14:46:15 -08001973func (library *libraryDecorator) BuildOnlyStatic() {
Colin Crossa48ab5b2017-02-14 15:28:44 -08001974 library.MutatedProperties.BuildShared = false
Colin Crossab3b7322016-12-09 14:46:15 -08001975}
1976
Chris Parsons3c27ca32020-11-20 12:42:07 -05001977// BuildOnlyShared disables building this library as a static library.
Colin Crossab3b7322016-12-09 14:46:15 -08001978func (library *libraryDecorator) BuildOnlyShared() {
Colin Crossa48ab5b2017-02-14 15:28:44 -08001979 library.MutatedProperties.BuildStatic = false
Colin Crossab3b7322016-12-09 14:46:15 -08001980}
1981
Chris Parsons3c27ca32020-11-20 12:42:07 -05001982// HeaderOnly disables building this library as a shared or static library;
1983// the library only exists to propagate header file dependencies up the build graph.
Colin Cross5950f382016-12-13 12:50:57 -08001984func (library *libraryDecorator) HeaderOnly() {
Colin Crossa48ab5b2017-02-14 15:28:44 -08001985 library.MutatedProperties.BuildShared = false
1986 library.MutatedProperties.BuildStatic = false
Colin Cross5950f382016-12-13 12:50:57 -08001987}
1988
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001989// HasLLNDKStubs returns true if this cc_library module has a variant that will build LLNDK stubs.
1990func (library *libraryDecorator) HasLLNDKStubs() bool {
Colin Cross203b4212021-04-26 17:19:41 -07001991 return String(library.Properties.Llndk.Symbol_file) != ""
Colin Cross0fb7fcd2021-03-02 11:00:07 -08001992}
1993
Colin Cross203b4212021-04-26 17:19:41 -07001994// hasLLNDKStubs returns true if this cc_library module has a variant that will build LLNDK stubs.
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001995func (library *libraryDecorator) HasLLNDKHeaders() bool {
Colin Cross1f3f1302021-04-26 18:37:44 -07001996 return Bool(library.Properties.Llndk.Llndk_headers)
1997}
1998
Ivan Lozano9eaacc82024-10-30 14:28:17 +00001999// IsLLNDKMovedToApex returns true if this cc_library module sets the llndk.moved_to_apex property.
2000func (library *libraryDecorator) IsLLNDKMovedToApex() bool {
Colin Cross0510f542024-11-13 16:07:08 -08002001 return Bool(library.Properties.Llndk.Moved_to_apex)
2002}
2003
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002004// HasVendorPublicLibrary returns true if this cc_library module has a variant that will build
Colin Cross5271fea2021-04-27 13:06:04 -07002005// vendor public library stubs.
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002006func (library *libraryDecorator) HasVendorPublicLibrary() bool {
Colin Cross5271fea2021-04-27 13:06:04 -07002007 return String(library.Properties.Vendor_public_library.Symbol_file) != ""
2008}
2009
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002010func (library *libraryDecorator) ImplementationModuleName(name string) string {
Colin Cross0477b422020-10-13 18:43:54 -07002011 return name
2012}
2013
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002014func (library *libraryDecorator) BuildStubs() bool {
Jiyong Park7ed9de32018-10-15 22:25:07 +09002015 return library.MutatedProperties.BuildStubs
2016}
2017
Jiyong Parka034b832019-08-27 14:02:19 +09002018func (library *libraryDecorator) symbolFileForAbiCheck(ctx ModuleContext) *string {
Colin Cross91ae5ec2024-10-01 14:03:40 -07002019 if props := library.getHeaderAbiCheckerProperties(ctx.Module().(*Module)); props.Symbol_file != nil {
Hsin-Yi Chen8feb4132022-12-26 15:54:36 +08002020 return props.Symbol_file
Jiyong Parka034b832019-08-27 14:02:19 +09002021 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002022 if library.HasStubsVariants() && library.Properties.Stubs.Symbol_file != nil {
Jiyong Parka034b832019-08-27 14:02:19 +09002023 return library.Properties.Stubs.Symbol_file
2024 }
Hsin-Yi Chenf1f276c2024-03-11 20:02:34 +08002025 // TODO(b/309880485): Distinguish platform, NDK, LLNDK, and APEX version scripts.
2026 if library.baseLinker.Properties.Version_script != nil {
2027 return library.baseLinker.Properties.Version_script
2028 }
Jiyong Parka034b832019-08-27 14:02:19 +09002029 return nil
2030}
2031
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002032func (library *libraryDecorator) HasStubsVariants() bool {
Jiyong Parkd4a3a132021-03-17 20:21:35 +09002033 // Just having stubs.symbol_file is enough to create a stub variant. In that case
2034 // the stub for the future API level is created.
2035 return library.Properties.Stubs.Symbol_file != nil ||
2036 len(library.Properties.Stubs.Versions) > 0
Colin Crossc88c2722020-09-28 17:32:47 -07002037}
2038
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002039func (library *libraryDecorator) IsStubsImplementationRequired() bool {
Alan Stokes73feba32022-11-14 12:21:24 +00002040 return BoolDefault(library.Properties.Stubs.Implementation_installable, true)
2041}
2042
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002043func (library *libraryDecorator) StubsVersions(ctx android.BaseModuleContext) []string {
2044 if !library.HasStubsVariants() {
Jiyong Parkd4a3a132021-03-17 20:21:35 +09002045 return nil
2046 }
2047
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002048 if library.HasLLNDKStubs() && ctx.Module().(*Module).InVendorOrProduct() {
Jooyung Han5e8994e2024-03-12 14:12:12 +09002049 // LLNDK libraries only need a single stubs variant (""), which is
2050 // added automatically in createVersionVariations().
2051 return nil
Colin Crossc1b36442021-05-06 13:42:48 -07002052 }
2053
Jiyong Parkd4a3a132021-03-17 20:21:35 +09002054 // Future API level is implicitly added if there isn't
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002055 versions := AddCurrentVersionIfNotPresent(library.Properties.Stubs.Versions)
2056 NormalizeVersions(ctx, versions)
Jooyung Han5e8994e2024-03-12 14:12:12 +09002057 return versions
Sam Delmerico75dbca22023-04-20 13:13:25 +00002058}
2059
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002060func AddCurrentVersionIfNotPresent(vers []string) []string {
Jiyong Parkd4a3a132021-03-17 20:21:35 +09002061 if inList(android.FutureApiLevel.String(), vers) {
2062 return vers
2063 }
2064 // In some cases, people use the raw value "10000" in the versions property.
2065 // We shouldn't add the future API level in that case, otherwise there will
2066 // be two identical versions.
2067 if inList(strconv.Itoa(android.FutureApiLevel.FinalOrFutureInt()), vers) {
2068 return vers
2069 }
2070 return append(vers, android.FutureApiLevel.String())
Colin Crossc88c2722020-09-28 17:32:47 -07002071}
2072
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002073func (library *libraryDecorator) SetStubsVersion(version string) {
Colin Crossc88c2722020-09-28 17:32:47 -07002074 library.MutatedProperties.StubsVersion = version
2075}
2076
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002077func (library *libraryDecorator) StubsVersion() string {
Jiyong Park7ed9de32018-10-15 22:25:07 +09002078 return library.MutatedProperties.StubsVersion
2079}
2080
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002081func (library *libraryDecorator) SetBuildStubs(isLatest bool) {
Colin Crossc88c2722020-09-28 17:32:47 -07002082 library.MutatedProperties.BuildStubs = true
Jiyong Parkd4a3a132021-03-17 20:21:35 +09002083 library.MutatedProperties.IsLatestVersion = isLatest
Colin Crossc88c2722020-09-28 17:32:47 -07002084}
2085
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002086func (library *libraryDecorator) SetAllStubsVersions(versions []string) {
Colin Crossc88c2722020-09-28 17:32:47 -07002087 library.MutatedProperties.AllStubsVersions = versions
2088}
2089
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002090func (library *libraryDecorator) AllStubsVersions() []string {
Colin Crossc88c2722020-09-28 17:32:47 -07002091 return library.MutatedProperties.AllStubsVersions
2092}
2093
Jooyung Hanad4c1872020-02-27 17:56:44 +09002094func (library *libraryDecorator) isLatestStubVersion() bool {
Jiyong Parkd4a3a132021-03-17 20:21:35 +09002095 return library.MutatedProperties.IsLatestVersion
Jooyung Hanad4c1872020-02-27 17:56:44 +09002096}
2097
Yu Liub73c3a62024-12-10 00:58:06 +00002098func (library *libraryDecorator) apexAvailable() []string {
Jiyong Parka90ca002019-10-07 15:47:24 +09002099 var list []string
2100 if library.static() {
2101 list = library.StaticProperties.Static.Apex_available
2102 } else if library.shared() {
2103 list = library.SharedProperties.Shared.Apex_available
2104 }
Yu Liub73c3a62024-12-10 00:58:06 +00002105
2106 return list
Jiyong Parka90ca002019-10-07 15:47:24 +09002107}
2108
Colin Cross1bc94122021-10-28 13:25:54 -07002109func (library *libraryDecorator) installable() *bool {
2110 if library.static() {
2111 return library.StaticProperties.Static.Installable
2112 } else if library.shared() {
2113 return library.SharedProperties.Shared.Installable
2114 }
2115 return nil
2116}
2117
Jingwen Chen8ac7d7d2023-03-20 11:05:16 +00002118func (library *libraryDecorator) makeUninstallable(mod *Module) {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002119 if library.static() && library.buildStatic() && !library.BuildStubs() {
Jingwen Chen8ac7d7d2023-03-20 11:05:16 +00002120 // If we're asked to make a static library uninstallable we don't do
2121 // anything since AndroidMkEntries always sets LOCAL_UNINSTALLABLE_MODULE
2122 // for these entries. This is done to still get the make targets for NOTICE
2123 // files from notice_files.mk, which other libraries might depend on.
2124 return
2125 }
2126 mod.ModuleBase.MakeUninstallable()
2127}
2128
Jihoon Kangf78a8902022-09-01 22:47:07 +00002129func (library *libraryDecorator) getPartition() string {
2130 return library.path.Partition()
2131}
2132
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002133func (library *libraryDecorator) GetAPIListCoverageXMLPath() android.ModuleOutPath {
Colin Crossceaa5322021-09-28 16:37:50 -07002134 return library.apiListCoverageXmlPath
2135}
2136
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002137func (library *libraryDecorator) setAPIListCoverageXMLPath(xml android.ModuleOutPath) {
2138 library.apiListCoverageXmlPath = xml
2139}
2140
Inseob Kima1888ce2022-10-04 14:42:02 +09002141func (library *libraryDecorator) overriddenModules() []string {
2142 return library.Properties.Overrides
2143}
2144
Cole Faustd143f3e2025-02-24 16:18:18 -08002145func (library *libraryDecorator) defaultDistFiles() []android.Path {
2146 if library.defaultDistFile == nil {
2147 return nil
2148 }
2149 return []android.Path{library.defaultDistFile}
2150}
2151
Inseob Kima1888ce2022-10-04 14:42:02 +09002152var _ overridable = (*libraryDecorator)(nil)
2153
Colin Cross571cccf2019-02-04 11:22:08 -08002154var versioningMacroNamesListKey = android.NewOnceKey("versioningMacroNamesList")
2155
Chris Parsons3c27ca32020-11-20 12:42:07 -05002156// versioningMacroNamesList returns a singleton map, where keys are "version macro names",
2157// and values are the module name responsible for registering the version macro name.
2158//
2159// Version macros are used when building against stubs, to provide version information about
2160// the stub. Only stub libraries should have an entry in this list.
2161//
2162// For example, when building against libFoo#ver, __LIBFOO_API__ macro is set to ver so
2163// that headers from libFoo can be conditionally compiled (this may hide APIs
2164// that are not available for the version).
2165//
2166// This map is used to ensure that there aren't conflicts between these version macro names.
Jiyong Parkda732bd2018-11-02 18:23:15 +09002167func versioningMacroNamesList(config android.Config) *map[string]string {
Colin Cross571cccf2019-02-04 11:22:08 -08002168 return config.Once(versioningMacroNamesListKey, func() interface{} {
Jiyong Parkda732bd2018-11-02 18:23:15 +09002169 m := make(map[string]string)
2170 return &m
2171 }).(*map[string]string)
2172}
2173
2174// alphanumeric and _ characters are preserved.
2175// other characters are all converted to _
2176var charsNotForMacro = regexp.MustCompile("[^a-zA-Z0-9_]+")
2177
Chris Parsons3c27ca32020-11-20 12:42:07 -05002178// versioningMacroName returns the canonical version macro name for the given module.
Jiyong Parkda732bd2018-11-02 18:23:15 +09002179func versioningMacroName(moduleName string) string {
2180 macroName := charsNotForMacro.ReplaceAllString(moduleName, "_")
Jooyung Hanb04a4992020-03-13 18:57:35 +09002181 macroName = strings.ToUpper(macroName)
Jiyong Parkda732bd2018-11-02 18:23:15 +09002182 return "__" + macroName + "_API__"
2183}
2184
Chris Parsons3c27ca32020-11-20 12:42:07 -05002185// NewLibrary builds and returns a new Module corresponding to a C++ library.
2186// Individual module implementations which comprise a C++ library (or something like
2187// a C++ library) should call this function, set some fields on the result, and
2188// then call the Init function.
Colin Crossab3b7322016-12-09 14:46:15 -08002189func NewLibrary(hod android.HostOrDeviceSupported) (*Module, *libraryDecorator) {
Colin Cross4d9c2d12016-07-29 12:48:20 -07002190 module := newModule(hod, android.MultilibBoth)
2191
Colin Crossb916a382016-07-29 17:28:03 -07002192 library := &libraryDecorator{
Colin Crossa48ab5b2017-02-14 15:28:44 -08002193 MutatedProperties: LibraryMutatedProperties{
Colin Crossab3b7322016-12-09 14:46:15 -08002194 BuildShared: true,
2195 BuildStatic: true,
Colin Cross4d9c2d12016-07-29 12:48:20 -07002196 },
Colin Crossb916a382016-07-29 17:28:03 -07002197 baseCompiler: NewBaseCompiler(),
Dan Albert61f32122018-07-26 14:00:24 -07002198 baseLinker: NewBaseLinker(module.sanitize),
Colin Crossb916a382016-07-29 17:28:03 -07002199 baseInstaller: NewBaseInstaller("lib", "lib64", InstallInSystem),
Jayant Chowdhary3e231fd2017-02-08 13:45:53 -08002200 sabi: module.sabi,
Colin Cross4d9c2d12016-07-29 12:48:20 -07002201 }
2202
Colin Crossb916a382016-07-29 17:28:03 -07002203 module.compiler = library
2204 module.linker = library
2205 module.installer = library
Colin Cross31076b32020-10-23 17:22:06 -07002206 module.library = library
Colin Crossb916a382016-07-29 17:28:03 -07002207
2208 return module, library
2209}
2210
Colin Cross10d22312017-05-03 11:01:58 -07002211// connects a shared library to a static library in order to reuse its .o files to avoid
2212// compiling source files twice.
Colin Cross767819f2024-05-22 14:22:34 -07002213func reuseStaticLibrary(ctx android.BottomUpMutatorContext, shared *Module) {
2214 if sharedCompiler, ok := shared.compiler.(*libraryDecorator); ok {
Dan Willemsen3a26eef2018-12-03 15:25:46 -08002215
2216 // Check libraries in addition to cflags, since libraries may be exporting different
2217 // include directories.
Colin Cross767819f2024-05-22 14:22:34 -07002218 if len(sharedCompiler.StaticProperties.Static.Cflags.GetOrDefault(ctx, nil)) == 0 &&
2219 len(sharedCompiler.SharedProperties.Shared.Cflags.GetOrDefault(ctx, nil)) == 0 &&
Cole Faustf0006e72024-08-19 14:39:19 -07002220 len(sharedCompiler.StaticProperties.Static.Whole_static_libs.GetOrDefault(ctx, nil)) == 0 &&
2221 len(sharedCompiler.SharedProperties.Shared.Whole_static_libs.GetOrDefault(ctx, nil)) == 0 &&
2222 len(sharedCompiler.StaticProperties.Static.Static_libs.GetOrDefault(ctx, nil)) == 0 &&
2223 len(sharedCompiler.SharedProperties.Shared.Static_libs.GetOrDefault(ctx, nil)) == 0 &&
2224 len(sharedCompiler.StaticProperties.Static.Shared_libs.GetOrDefault(ctx, nil)) == 0 &&
2225 len(sharedCompiler.SharedProperties.Shared.Shared_libs.GetOrDefault(ctx, nil)) == 0 &&
Martin Stjernholm10566a02020-03-24 01:19:52 +00002226 // Compare System_shared_libs properties with nil because empty lists are
2227 // semantically significant for them.
Colin Cross767819f2024-05-22 14:22:34 -07002228 sharedCompiler.StaticProperties.Static.System_shared_libs == nil &&
Colin Crosse1bb5d02019-09-24 14:55:04 -07002229 sharedCompiler.SharedProperties.Shared.System_shared_libs == nil {
Colin Cross10d22312017-05-03 11:01:58 -07002230
Colin Cross767819f2024-05-22 14:22:34 -07002231 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, reuseObjTag, ctx.ModuleName())
Colin Cross10d22312017-05-03 11:01:58 -07002232 }
Colin Cross0de8a1e2020-09-18 14:15:30 -07002233
2234 // This dep is just to reference static variant from shared variant
Colin Cross767819f2024-05-22 14:22:34 -07002235 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticVariantTag, ctx.ModuleName())
Colin Cross10d22312017-05-03 11:01:58 -07002236 }
2237}
2238
Colin Cross767819f2024-05-22 14:22:34 -07002239// linkageTransitionMutator adds "static" or "shared" variants for modules depending
Chris Parsons3c27ca32020-11-20 12:42:07 -05002240// on whether the module can be built as a static library or a shared library.
Colin Cross767819f2024-05-22 14:22:34 -07002241type linkageTransitionMutator struct{}
2242
2243func (linkageTransitionMutator) Split(ctx android.BaseModuleContext) []string {
Jaewoong Jung18aefc12020-12-21 09:11:10 -08002244 ccPrebuilt := false
Colin Cross767819f2024-05-22 14:22:34 -07002245 if m, ok := ctx.Module().(*Module); ok && m.linker != nil {
Jaewoong Jung18aefc12020-12-21 09:11:10 -08002246 _, ccPrebuilt = m.linker.(prebuiltLibraryInterface)
Ivan Lozano52767be2019-10-18 14:49:46 -07002247 }
Jaewoong Jung18aefc12020-12-21 09:11:10 -08002248 if ccPrebuilt {
Colin Cross767819f2024-05-22 14:22:34 -07002249 library := ctx.Module().(*Module).linker.(prebuiltLibraryInterface)
Colin Cross33b2fb72019-05-14 14:07:01 -07002250
Paul Duffinf5ea9e12020-02-21 10:57:00 +00002251 // Differentiate between header only and building an actual static/shared library
Colin Cross127bb8b2020-12-16 16:46:01 -08002252 buildStatic := library.buildStatic()
2253 buildShared := library.buildShared()
2254 if buildStatic || buildShared {
Paul Duffinf5ea9e12020-02-21 10:57:00 +00002255 // Always create both the static and shared variants for prebuilt libraries, and then disable the one
2256 // that is not being used. This allows them to share the name of a cc_library module, which requires that
2257 // all the variants of the cc_library also exist on the prebuilt.
Colin Cross767819f2024-05-22 14:22:34 -07002258 return []string{"static", "shared"}
Paul Duffinf5ea9e12020-02-21 10:57:00 +00002259 } else {
2260 // Header only
Colin Crossb916a382016-07-29 17:28:03 -07002261 }
Ivan Lozano806efd32024-12-11 21:38:53 +00002262 } else if library, ok := ctx.Module().(LinkableInterface); ok && (library.CcLibraryInterface()) {
Ivan Lozano2b262972019-11-21 12:30:50 -08002263 // Non-cc.Modules may need an empty variant for their mutators.
2264 variations := []string{}
2265 if library.NonCcVariants() {
2266 variations = append(variations, "")
2267 }
Colin Cross127bb8b2020-12-16 16:46:01 -08002268 isLLNDK := false
Colin Cross767819f2024-05-22 14:22:34 -07002269 if m, ok := ctx.Module().(*Module); ok {
Colin Cross203b4212021-04-26 17:19:41 -07002270 isLLNDK = m.IsLlndk()
Colin Cross127bb8b2020-12-16 16:46:01 -08002271 }
2272 buildStatic := library.BuildStaticVariant() && !isLLNDK
2273 buildShared := library.BuildSharedVariant()
2274 if buildStatic && buildShared {
Colin Cross767819f2024-05-22 14:22:34 -07002275 variations = append([]string{"static", "shared"}, variations...)
2276 return variations
Colin Cross127bb8b2020-12-16 16:46:01 -08002277 } else if buildStatic {
Colin Cross767819f2024-05-22 14:22:34 -07002278 variations = append([]string{"static"}, variations...)
Colin Cross127bb8b2020-12-16 16:46:01 -08002279 } else if buildShared {
Colin Cross767819f2024-05-22 14:22:34 -07002280 variations = append([]string{"shared"}, variations...)
Ivan Lozano2b262972019-11-21 12:30:50 -08002281 }
Ivan Lozano52767be2019-10-18 14:49:46 -07002282
Colin Cross767819f2024-05-22 14:22:34 -07002283 if len(variations) > 0 {
2284 return variations
2285 }
2286 }
2287 return []string{""}
2288}
2289
2290func (linkageTransitionMutator) OutgoingTransition(ctx android.OutgoingTransitionContext, sourceVariation string) string {
Colin Crossd8d8b852024-12-20 16:32:37 -08002291 if ctx.DepTag() == android.PrebuiltDepTag {
2292 return sourceVariation
2293 }
Colin Cross767819f2024-05-22 14:22:34 -07002294 return ""
2295}
2296
2297func (linkageTransitionMutator) IncomingTransition(ctx android.IncomingTransitionContext, incomingVariation string) string {
2298 ccPrebuilt := false
2299 if m, ok := ctx.Module().(*Module); ok && m.linker != nil {
2300 _, ccPrebuilt = m.linker.(prebuiltLibraryInterface)
2301 }
2302 if ccPrebuilt {
2303 if incomingVariation != "" {
2304 return incomingVariation
2305 }
2306 library := ctx.Module().(*Module).linker.(prebuiltLibraryInterface)
2307 if library.buildShared() {
2308 return "shared"
2309 } else if library.buildStatic() {
2310 return "static"
2311 }
2312 return ""
2313 } else if library, ok := ctx.Module().(LinkableInterface); ok && library.CcLibraryInterface() {
2314 isLLNDK := false
2315 if m, ok := ctx.Module().(*Module); ok {
2316 isLLNDK = m.IsLlndk()
2317 }
2318 buildStatic := library.BuildStaticVariant() && !isLLNDK
2319 buildShared := library.BuildSharedVariant()
Ivan Lozano806efd32024-12-11 21:38:53 +00002320 if library.BuildRlibVariant() && !buildStatic && (incomingVariation == "static" || incomingVariation == "") {
Ivan Lozanofd47b1a2024-05-17 14:13:41 -04002321 // Rust modules do not build static libs, but rlibs are used as if they
2322 // were via `static_libs`. Thus we need to alias the BuildRlibVariant
2323 // to "static" for Rust FFI libraries.
Colin Cross767819f2024-05-22 14:22:34 -07002324 return ""
2325 }
2326 if incomingVariation != "" {
2327 return incomingVariation
2328 }
2329 if buildShared {
2330 return "shared"
2331 } else if buildStatic {
2332 return "static"
2333 }
2334 return ""
2335 }
2336 return ""
2337}
2338
2339func (linkageTransitionMutator) Mutate(ctx android.BottomUpMutatorContext, variation string) {
2340 ccPrebuilt := false
2341 if m, ok := ctx.Module().(*Module); ok && m.linker != nil {
2342 _, ccPrebuilt = m.linker.(prebuiltLibraryInterface)
2343 }
2344 if ccPrebuilt {
2345 library := ctx.Module().(*Module).linker.(prebuiltLibraryInterface)
2346 if variation == "static" {
2347 library.setStatic()
2348 if !library.buildStatic() {
2349 library.disablePrebuilt()
Colin Crossd8d8b852024-12-20 16:32:37 -08002350 ctx.Module().(*Module).Prebuilt().SetUsePrebuilt(false)
Colin Cross767819f2024-05-22 14:22:34 -07002351 }
2352 } else if variation == "shared" {
2353 library.setShared()
2354 if !library.buildShared() {
2355 library.disablePrebuilt()
Colin Crossd8d8b852024-12-20 16:32:37 -08002356 ctx.Module().(*Module).Prebuilt().SetUsePrebuilt(false)
Colin Cross767819f2024-05-22 14:22:34 -07002357 }
2358 }
2359 } else if library, ok := ctx.Module().(LinkableInterface); ok && library.CcLibraryInterface() {
2360 if variation == "static" {
2361 library.SetStatic()
2362 } else if variation == "shared" {
2363 library.SetShared()
2364 var isLLNDK bool
2365 if m, ok := ctx.Module().(*Module); ok {
2366 isLLNDK = m.IsLlndk()
2367 }
2368 buildStatic := library.BuildStaticVariant() && !isLLNDK
2369 buildShared := library.BuildSharedVariant()
2370 if buildStatic && buildShared {
2371 if _, ok := library.(*Module); ok {
2372 reuseStaticLibrary(ctx, library.(*Module))
2373 }
2374 }
Colin Cross571cccf2019-02-04 11:22:08 -08002375 }
2376 }
Jiyong Park25fc6a92018-11-18 18:02:45 +09002377}
2378
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002379// NormalizeVersions modifies `versions` in place, so that each raw version
Chris Parsons3c27ca32020-11-20 12:42:07 -05002380// string becomes its normalized canonical form.
2381// Validates that the versions in `versions` are specified in least to greatest order.
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002382func NormalizeVersions(ctx android.BaseModuleContext, versions []string) {
Dan Albertc8060532020-07-22 22:32:17 -07002383 var previous android.ApiLevel
Jooyung Hanad4c1872020-02-27 17:56:44 +09002384 for i, v := range versions {
Dan Albertc8060532020-07-22 22:32:17 -07002385 ver, err := android.ApiLevelFromUser(ctx, v)
Jooyung Hanad4c1872020-02-27 17:56:44 +09002386 if err != nil {
Jooyung Hanaed150d2020-04-02 01:41:41 +09002387 ctx.PropertyErrorf("versions", "%s", err.Error())
2388 return
Jooyung Hanad4c1872020-02-27 17:56:44 +09002389 }
Dan Albertc8060532020-07-22 22:32:17 -07002390 if i > 0 && ver.LessThanOrEqualTo(previous) {
2391 ctx.PropertyErrorf("versions", "not sorted: %v", versions)
2392 }
2393 versions[i] = ver.String()
2394 previous = ver
Jooyung Hanaed150d2020-04-02 01:41:41 +09002395 }
Jooyung Hanad4c1872020-02-27 17:56:44 +09002396}
2397
Colin Crossadd04a82024-05-22 09:57:59 -07002398func perApiVersionVariations(mctx android.BaseModuleContext, minSdkVersion string) []string {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002399 from, err := NativeApiLevelFromUser(mctx, minSdkVersion)
Colin Crossbbc941b2020-09-30 12:27:01 -07002400 if err != nil {
2401 mctx.PropertyErrorf("min_sdk_version", err.Error())
Colin Crossadd04a82024-05-22 09:57:59 -07002402 return []string{""}
Colin Crossbbc941b2020-09-30 12:27:01 -07002403 }
2404
Colin Crossadd04a82024-05-22 09:57:59 -07002405 return ndkLibraryVersions(mctx, from)
Colin Crossbbc941b2020-09-30 12:27:01 -07002406}
2407
Liz Kammer23942242022-04-08 15:41:00 -04002408func canBeOrLinkAgainstVersionVariants(module interface {
Jooyung Han624d35c2020-04-10 12:57:24 +09002409 Host() bool
2410 InRamdisk() bool
Yifan Hong60e0cfb2020-10-21 15:17:56 -07002411 InVendorRamdisk() bool
Jooyung Han624d35c2020-04-10 12:57:24 +09002412}) bool {
Jose Galmes6f843bc2020-12-11 13:36:29 -08002413 return !module.Host() && !module.InRamdisk() && !module.InVendorRamdisk()
Jooyung Han624d35c2020-04-10 12:57:24 +09002414}
2415
Liz Kammer23942242022-04-08 15:41:00 -04002416func canBeVersionVariant(module interface {
Colin Cross3146c5c2020-09-30 15:34:40 -07002417 Host() bool
2418 InRamdisk() bool
Yifan Hong60e0cfb2020-10-21 15:17:56 -07002419 InVendorRamdisk() bool
Colin Cross3146c5c2020-09-30 15:34:40 -07002420 CcLibraryInterface() bool
2421 Shared() bool
Colin Cross3146c5c2020-09-30 15:34:40 -07002422}) bool {
Liz Kammer23942242022-04-08 15:41:00 -04002423 return canBeOrLinkAgainstVersionVariants(module) &&
Colin Crossa717db72020-10-23 14:53:06 -07002424 module.CcLibraryInterface() && module.Shared()
Colin Cross3146c5c2020-09-30 15:34:40 -07002425}
2426
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002427func moduleVersionedInterface(module blueprint.Module) VersionedInterface {
2428 if m, ok := module.(VersionedLinkableInterface); ok {
2429 return m.VersionedInterface()
Colin Cross31076b32020-10-23 17:22:06 -07002430 }
2431 return nil
2432}
2433
Liz Kammer23942242022-04-08 15:41:00 -04002434// setStubsVersions normalizes the versions in the Stubs.Versions property into MutatedProperties.AllStubsVersions.
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002435func setStubsVersions(mctx android.BaseModuleContext, module VersionedLinkableInterface) {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002436 if !module.BuildSharedVariant() || !canBeVersionVariant(module) {
Liz Kammer23942242022-04-08 15:41:00 -04002437 return
Colin Crossd1f898e2020-08-18 18:35:15 -07002438 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002439 versions := module.VersionedInterface().StubsVersions(mctx)
Liz Kammer23942242022-04-08 15:41:00 -04002440 if mctx.Failed() {
2441 return
2442 }
2443 // Set the versions on the pre-mutated module so they can be read by any llndk modules that
2444 // depend on the implementation library and haven't been mutated yet.
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002445 module.VersionedInterface().SetAllStubsVersions(versions)
Colin Crossd1f898e2020-08-18 18:35:15 -07002446}
Jooyung Han61b66e92020-03-21 14:21:46 +00002447
Colin Crossadd04a82024-05-22 09:57:59 -07002448// versionTransitionMutator splits a module into the mandatory non-stubs variant
Colin Crossd1f898e2020-08-18 18:35:15 -07002449// (which is unnamed) and zero or more stubs variants.
Colin Crossadd04a82024-05-22 09:57:59 -07002450type versionTransitionMutator struct{}
2451
2452func (versionTransitionMutator) Split(ctx android.BaseModuleContext) []string {
2453 if ctx.Os() != android.Android {
2454 return []string{""}
Liz Kammer23942242022-04-08 15:41:00 -04002455 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002456 if m, ok := ctx.Module().(VersionedLinkableInterface); ok {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002457 if m.CcLibraryInterface() && canBeVersionVariant(m) {
2458 setStubsVersions(ctx, m)
2459 return append(slices.Clone(m.VersionedInterface().AllStubsVersions()), "")
2460 } else if m.SplitPerApiLevel() && m.IsSdkVariant() {
2461 return perApiVersionVariations(ctx, m.MinSdkVersion())
2462 }
Colin Crossbbc941b2020-09-30 12:27:01 -07002463 }
2464
Colin Crossadd04a82024-05-22 09:57:59 -07002465 return []string{""}
2466}
2467
2468func (versionTransitionMutator) OutgoingTransition(ctx android.OutgoingTransitionContext, sourceVariation string) string {
Colin Crossd8d8b852024-12-20 16:32:37 -08002469 if ctx.DepTag() == android.PrebuiltDepTag {
2470 return sourceVariation
2471 }
Colin Crossadd04a82024-05-22 09:57:59 -07002472 return ""
2473}
2474
2475func (versionTransitionMutator) IncomingTransition(ctx android.IncomingTransitionContext, incomingVariation string) string {
2476 if ctx.Os() != android.Android {
2477 return ""
2478 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002479 m, ok := ctx.Module().(VersionedLinkableInterface)
2480 if library := moduleVersionedInterface(ctx.Module()); library != nil && canBeVersionVariant(m) {
Colin Crossadd04a82024-05-22 09:57:59 -07002481 if incomingVariation == "latest" {
2482 latestVersion := ""
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002483 versions := library.AllStubsVersions()
Colin Crossadd04a82024-05-22 09:57:59 -07002484 if len(versions) > 0 {
2485 latestVersion = versions[len(versions)-1]
2486 }
2487 return latestVersion
Colin Crossbbc941b2020-09-30 12:27:01 -07002488 }
Colin Crossadd04a82024-05-22 09:57:59 -07002489 return incomingVariation
2490 } else if ok && m.SplitPerApiLevel() && m.IsSdkVariant() {
2491 // If this module only has variants with versions and the incoming dependency doesn't specify which one
2492 // is needed then assume the latest version.
2493 if incomingVariation == "" {
2494 return android.FutureApiLevel.String()
2495 }
2496 return incomingVariation
2497 }
2498
2499 return ""
2500}
2501
2502func (versionTransitionMutator) Mutate(ctx android.BottomUpMutatorContext, variation string) {
2503 // Optimization: return early if this module can't be affected.
2504 if ctx.Os() != android.Android {
2505 return
2506 }
2507
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002508 m, ok := ctx.Module().(VersionedLinkableInterface)
2509 if library := moduleVersionedInterface(ctx.Module()); library != nil && canBeVersionVariant(m) {
Colin Crossadd04a82024-05-22 09:57:59 -07002510 isLLNDK := m.IsLlndk()
2511 isVendorPublicLibrary := m.IsVendorPublicLibrary()
Colin Crossadd04a82024-05-22 09:57:59 -07002512
Spandan Dasff665182024-09-11 18:48:44 +00002513 if variation != "" || isLLNDK || isVendorPublicLibrary {
Colin Crossadd04a82024-05-22 09:57:59 -07002514 // A stubs or LLNDK stubs variant.
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002515 if sm, ok := ctx.Module().(PlatformSanitizeable); ok && sm.SanitizePropDefined() {
2516 sm.ForceDisableSanitizers()
Colin Crossadd04a82024-05-22 09:57:59 -07002517 }
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002518 m.SetStl("none")
2519 m.SetPreventInstall()
2520 allStubsVersions := m.VersionedInterface().AllStubsVersions()
Colin Crossadd04a82024-05-22 09:57:59 -07002521 isLatest := len(allStubsVersions) > 0 && variation == allStubsVersions[len(allStubsVersions)-1]
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002522 m.VersionedInterface().SetBuildStubs(isLatest)
Colin Crossadd04a82024-05-22 09:57:59 -07002523 }
2524 if variation != "" {
2525 // A non-LLNDK stubs module is hidden from make
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002526 m.VersionedInterface().SetStubsVersion(variation)
2527 m.SetHideFromMake()
Colin Crossadd04a82024-05-22 09:57:59 -07002528 } else {
2529 // A non-LLNDK implementation module has a dependency to all stubs versions
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002530 for _, version := range m.VersionedInterface().AllStubsVersions() {
2531 ctx.AddVariationDependencies(
2532 []blueprint.Variation{
2533 {Mutator: "version", Variation: version},
2534 {Mutator: "link", Variation: "shared"}},
2535 StubImplDepTag, ctx.ModuleName())
Colin Crossadd04a82024-05-22 09:57:59 -07002536 }
2537 }
2538 } else if ok && m.SplitPerApiLevel() && m.IsSdkVariant() {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002539 m.SetSdkVersion(variation)
2540 m.SetMinSdkVersion(variation)
Jiyong Park7ed9de32018-10-15 22:25:07 +09002541 }
Jiyong Park7ed9de32018-10-15 22:25:07 +09002542}
Colin Crossd7227f92019-09-05 14:26:33 -07002543
2544// maybeInjectBoringSSLHash adds a rule to run bssl_inject_hash on the output file if the module has the
2545// inject_bssl_hash or if any static library dependencies have inject_bssl_hash set. It returns the output path
2546// that the linked output file should be written to.
2547// TODO(b/137267623): Remove this in favor of a cc_genrule when they support operating on shared libraries.
2548func maybeInjectBoringSSLHash(ctx android.ModuleContext, outputFile android.ModuleOutPath,
2549 inject *bool, fileName string) android.ModuleOutPath {
2550 // TODO(b/137267623): Remove this in favor of a cc_genrule when they support operating on shared libraries.
2551 injectBoringSSLHash := Bool(inject)
Yu Liu116610a2025-01-06 21:54:48 +00002552 ctx.VisitDirectDepsProxy(func(dep android.ModuleProxy) {
Colin Cross6e511a92020-07-27 21:26:48 -07002553 if tag, ok := ctx.OtherModuleDependencyTag(dep).(libraryDependencyTag); ok && tag.static() {
Ivan Lozano9eaacc82024-10-30 14:28:17 +00002554 if ccInfo, ok := android.OtherModuleProvider(ctx, dep, CcInfoProvider); ok &&
2555 ccInfo.LinkerInfo != nil && ccInfo.LinkerInfo.LibraryDecoratorInfo != nil {
Yu Liu116610a2025-01-06 21:54:48 +00002556 if ccInfo.LinkerInfo.LibraryDecoratorInfo.InjectBsslHash {
2557 injectBoringSSLHash = true
Colin Crossd7227f92019-09-05 14:26:33 -07002558 }
2559 }
2560 }
2561 })
2562 if injectBoringSSLHash {
2563 hashedOutputfile := outputFile
2564 outputFile = android.PathForModuleOut(ctx, "unhashed", fileName)
2565
Colin Crossf1a035e2020-11-16 17:32:30 -08002566 rule := android.NewRuleBuilder(pctx, ctx)
Colin Crossd7227f92019-09-05 14:26:33 -07002567 rule.Command().
Colin Crossf1a035e2020-11-16 17:32:30 -08002568 BuiltTool("bssl_inject_hash").
Colin Crossd7227f92019-09-05 14:26:33 -07002569 FlagWithInput("-in-object ", outputFile).
2570 FlagWithOutput("-o ", hashedOutputfile)
Colin Crossf1a035e2020-11-16 17:32:30 -08002571 rule.Build("injectCryptoHash", "inject crypto hash")
Colin Crossd7227f92019-09-05 14:26:33 -07002572 }
2573
2574 return outputFile
2575}