blob: 9d4a643d2fb00ad47d0c99700fd2c6e47158a8f7 [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
Colin Crossb916a382016-07-29 17:28:03 -070017import (
18 "android/soong/android"
Yi Kong7df0f302018-10-08 22:10:12 +000019 "android/soong/cc/config"
Colin Crossb916a382016-07-29 17:28:03 -070020 "fmt"
Zhizhou Yang9100b1d2018-11-30 14:00:04 -080021 "strconv"
Colin Cross4b963f82016-09-29 14:06:02 -070022
Colin Cross86803cf2018-02-15 14:12:26 -080023 "github.com/google/blueprint"
Colin Cross4b963f82016-09-29 14:06:02 -070024 "github.com/google/blueprint/proptools"
Colin Crossb916a382016-07-29 17:28:03 -070025)
26
Colin Cross4d9c2d12016-07-29 12:48:20 -070027// This file contains the basic functionality for linking against static libraries and shared
28// libraries. Final linking into libraries or executables is handled in library.go, binary.go, etc.
29
30type BaseLinkerProperties struct {
31 // list of modules whose object files should be linked into this module
32 // in their entirety. For static library modules, all of the .o files from the intermediate
33 // directory of the dependency will be linked into this modules .a file. For a shared library,
34 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
35 Whole_static_libs []string `android:"arch_variant,variant_prepend"`
36
37 // list of modules that should be statically linked into this module.
38 Static_libs []string `android:"arch_variant,variant_prepend"`
39
40 // list of modules that should be dynamically linked into this module.
41 Shared_libs []string `android:"arch_variant"`
42
Colin Cross5950f382016-12-13 12:50:57 -080043 // list of modules that should only provide headers for this module.
44 Header_libs []string `android:"arch_variant,variant_prepend"`
45
Colin Cross4d9c2d12016-07-29 12:48:20 -070046 // list of module-specific flags that will be used for all link steps
47 Ldflags []string `android:"arch_variant"`
48
Colin Cross4d9c2d12016-07-29 12:48:20 -070049 // list of system libraries that will be dynamically linked to
Colin Crossef88ae22017-08-18 16:52:25 -070050 // shared library and executable modules. If unset, generally defaults to libc,
51 // libm, and libdl. Set to [] to prevent linking against the defaults.
Dan Willemsen3a26eef2018-12-03 15:25:46 -080052 System_shared_libs []string `android:"arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070053
54 // allow the module to contain undefined symbols. By default,
55 // modules cannot contain undefined symbols that are not satisified by their immediate
56 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
57 // This flag should only be necessary for compiling low-level libraries like libc.
Colin Crossbe360ae2016-12-08 09:45:21 -080058 Allow_undefined_symbols *bool `android:"arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070059
Yi Kong2c188be2018-10-10 13:43:02 -070060 // don't link in libclang_rt.builtins-*.a
Yi Kong410d8342019-02-15 10:07:18 -080061 No_libcrt *bool `android:"arch_variant"`
Yi Kong2c188be2018-10-10 13:43:02 -070062
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -070063 // Use clang lld instead of gnu ld.
Pirama Arumuga Nainar2b8959a2018-04-20 11:52:42 -070064 Use_clang_lld *bool `android:"arch_variant"`
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -070065
Colin Cross4d9c2d12016-07-29 12:48:20 -070066 // -l arguments to pass to linker for host-provided shared libraries
67 Host_ldlibs []string `android:"arch_variant"`
68
69 // list of shared libraries to re-export include directories from. Entries must be
70 // present in shared_libs.
71 Export_shared_lib_headers []string `android:"arch_variant"`
72
73 // list of static libraries to re-export include directories from. Entries must be
74 // present in static_libs.
75 Export_static_lib_headers []string `android:"arch_variant"`
76
Colin Cross5950f382016-12-13 12:50:57 -080077 // list of header libraries to re-export include directories from. Entries must be
78 // present in header_libs.
79 Export_header_lib_headers []string `android:"arch_variant"`
80
Dan Willemsenb3454ab2016-09-28 17:34:58 -070081 // list of generated headers to re-export include directories from. Entries must be
82 // present in generated_headers.
83 Export_generated_headers []string `android:"arch_variant"`
84
Colin Cross4d9c2d12016-07-29 12:48:20 -070085 // don't link in crt_begin and crt_end. This flag should only be necessary for
86 // compiling crt or libc.
87 Nocrt *bool `android:"arch_variant"`
Colin Cross18c0c5a2016-12-01 14:45:23 -080088
89 // group static libraries. This can resolve missing symbols issues with interdependencies
90 // between static libraries, but it is generally better to order them correctly instead.
91 Group_static_libs *bool `android:"arch_variant"`
Jiyong Park44cf1a72017-06-16 12:03:55 +090092
Logan Chien43d34c32017-12-20 01:17:32 +080093 // list of modules that should be installed with this module. This is similar to 'required'
94 // but '.vendor' suffix will be appended to the module names if the shared libraries have
95 // vendor variants and this module uses VNDK.
96 Runtime_libs []string `android:"arch_variant"`
97
Jiyong Park44cf1a72017-06-16 12:03:55 +090098 Target struct {
Justin Yun63e9ec72020-10-29 16:49:43 +090099 Vendor, Product struct {
100 // list of shared libs that only should be used to build vendor or
101 // product variant of the C/C++ module.
Martin Stjernholmef449fe2018-11-06 16:12:13 +0000102 Shared_libs []string
103
Justin Yun63e9ec72020-10-29 16:49:43 +0900104 // list of static libs that only should be used to build vendor or
105 // product variant of the C/C++ module.
Chong Zhanged5c1462020-02-20 15:04:35 -0800106 Static_libs []string
107
Justin Yun63e9ec72020-10-29 16:49:43 +0900108 // list of shared libs that should not be used to build vendor or
109 // product variant of the C/C++ module.
Jiyong Park44cf1a72017-06-16 12:03:55 +0900110 Exclude_shared_libs []string
Jiyong Park52d25bd2017-10-13 09:17:01 +0900111
Justin Yun63e9ec72020-10-29 16:49:43 +0900112 // list of static libs that should not be used to build vendor or
113 // product variant of the C/C++ module.
Jiyong Park52d25bd2017-10-13 09:17:01 +0900114 Exclude_static_libs []string
Logan Chien43d34c32017-12-20 01:17:32 +0800115
Justin Yun63e9ec72020-10-29 16:49:43 +0900116 // list of header libs that should not be used to build vendor or
117 // product variant of the C/C++ module.
Jiwen 'Steve' Cai8dbc6532018-08-08 10:58:34 -0700118 Exclude_header_libs []string
119
Justin Yun63e9ec72020-10-29 16:49:43 +0900120 // list of runtime libs that should not be installed along with
121 // vendor or variant of the C/C++ module.
Logan Chien43d34c32017-12-20 01:17:32 +0800122 Exclude_runtime_libs []string
Colin Crossc17727d2018-10-24 12:42:09 -0700123
Justin Yun63e9ec72020-10-29 16:49:43 +0900124 // version script for vendor or product variant
Colin Crossc17727d2018-10-24 12:42:09 -0700125 Version_script *string `android:"arch_variant"`
Jiyong Park44cf1a72017-06-16 12:03:55 +0900126 }
Jiyong Parkf9332f12018-02-01 00:54:12 +0900127 Recovery struct {
Martin Stjernholmef449fe2018-11-06 16:12:13 +0000128 // list of shared libs that only should be used to build the recovery
129 // variant of the C/C++ module.
130 Shared_libs []string
131
Yifan Honge7efbc82019-08-01 13:45:42 -0700132 // list of static libs that only should be used to build the recovery
133 // variant of the C/C++ module.
134 Static_libs []string
135
Jiyong Parkf9332f12018-02-01 00:54:12 +0900136 // list of shared libs that should not be used to build
137 // the recovery variant of the C/C++ module.
138 Exclude_shared_libs []string
139
140 // list of static libs that should not be used to build
141 // the recovery variant of the C/C++ module.
142 Exclude_static_libs []string
Hridya Valsaraju280febf2018-08-10 09:08:13 -0700143
144 // list of header libs that should not be used to build the recovery variant
145 // of the C/C++ module.
146 Exclude_header_libs []string
Jiyong Parkf9332f12018-02-01 00:54:12 +0900147 }
Yifan Hongcf4832c2020-01-21 17:04:13 -0800148 Ramdisk struct {
149 // list of static libs that only should be used to build the recovery
150 // variant of the C/C++ module.
151 Static_libs []string
152
153 // list of shared libs that should not be used to build
154 // the ramdisk variant of the C/C++ module.
155 Exclude_shared_libs []string
156
157 // list of static libs that should not be used to build
158 // the ramdisk variant of the C/C++ module.
159 Exclude_static_libs []string
160 }
Yifan Hong6da33c22020-10-27 15:01:21 -0700161 Vendor_ramdisk struct {
162 // list of shared libs that should not be used to build
163 // the recovery variant of the C/C++ module.
164 Exclude_shared_libs []string
165
166 // list of static libs that should not be used to build
167 // the vendor ramdisk variant of the C/C++ module.
168 Exclude_static_libs []string
169 }
Colin Crossc511bc52020-04-07 16:50:32 +0000170 Platform struct {
171 // list of shared libs that should be use to build the platform variant
172 // of a module that sets sdk_version. This should rarely be necessary,
173 // in most cases the same libraries are available for the SDK and platform
174 // variants.
175 Shared_libs []string
176 }
Jiyong Park44cf1a72017-06-16 12:03:55 +0900177 }
Colin Cross86803cf2018-02-15 14:12:26 -0800178
179 // make android::build:GetBuildNumber() available containing the build ID.
180 Use_version_lib *bool `android:"arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -0700181
Chih-Hung Hsieh86814712018-05-23 18:30:46 -0700182 // Generate compact dynamic relocation table, default true.
183 Pack_relocations *bool `android:"arch_variant"`
Dan Albert61f32122018-07-26 14:00:24 -0700184
185 // local file name to pass to the linker as --version_script
Colin Cross27b922f2019-03-04 22:35:41 -0800186 Version_script *string `android:"path,arch_variant"`
Christopher Ferrisc71193a2019-12-16 09:55:10 -0800187
188 // list of static libs that should not be used to build this module
Colin Cross06210312020-06-10 15:47:34 -0700189 Exclude_static_libs []string `android:"arch_variant"`
190
191 // list of shared libs that should not be used to build this module
192 Exclude_shared_libs []string `android:"arch_variant"`
Chih-Hung Hsieh86814712018-05-23 18:30:46 -0700193}
194
Dan Albert61f32122018-07-26 14:00:24 -0700195func NewBaseLinker(sanitize *sanitize) *baseLinker {
196 return &baseLinker{sanitize: sanitize}
Colin Crossb916a382016-07-29 17:28:03 -0700197}
198
Colin Cross4d9c2d12016-07-29 12:48:20 -0700199// baseLinker provides support for shared_libs, static_libs, and whole_static_libs properties
200type baseLinker struct {
201 Properties BaseLinkerProperties
202 dynamicProperties struct {
Jiyong Park7ed9de32018-10-15 22:25:07 +0900203 RunPaths []string `blueprint:"mutated"`
204 BuildStubs bool `blueprint:"mutated"`
Colin Cross4d9c2d12016-07-29 12:48:20 -0700205 }
Dan Albert61f32122018-07-26 14:00:24 -0700206
207 sanitize *sanitize
Colin Cross4d9c2d12016-07-29 12:48:20 -0700208}
209
210func (linker *baseLinker) appendLdflags(flags []string) {
211 linker.Properties.Ldflags = append(linker.Properties.Ldflags, flags...)
212}
213
Chris Parsonse0f2ab32020-11-20 17:27:25 -0500214// linkerInit initializes dynamic properties of the linker (such as runpath).
Colin Cross42742b82016-08-01 13:20:05 -0700215func (linker *baseLinker) linkerInit(ctx BaseModuleContext) {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700216 if ctx.toolchain().Is64Bit() {
Colin Crossb916a382016-07-29 17:28:03 -0700217 linker.dynamicProperties.RunPaths = append(linker.dynamicProperties.RunPaths, "../lib64", "lib64")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700218 } else {
Colin Crossb916a382016-07-29 17:28:03 -0700219 linker.dynamicProperties.RunPaths = append(linker.dynamicProperties.RunPaths, "../lib", "lib")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700220 }
221}
222
Colin Cross42742b82016-08-01 13:20:05 -0700223func (linker *baseLinker) linkerProps() []interface{} {
Colin Crossc17727d2018-10-24 12:42:09 -0700224 return []interface{}{&linker.Properties, &linker.dynamicProperties}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700225}
226
Dan Albert61f32122018-07-26 14:00:24 -0700227func (linker *baseLinker) linkerDeps(ctx DepsContext, deps Deps) Deps {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700228 deps.WholeStaticLibs = append(deps.WholeStaticLibs, linker.Properties.Whole_static_libs...)
Colin Cross5950f382016-12-13 12:50:57 -0800229 deps.HeaderLibs = append(deps.HeaderLibs, linker.Properties.Header_libs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700230 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Static_libs...)
231 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Shared_libs...)
Logan Chien43d34c32017-12-20 01:17:32 +0800232 deps.RuntimeLibs = append(deps.RuntimeLibs, linker.Properties.Runtime_libs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700233
Colin Cross5950f382016-12-13 12:50:57 -0800234 deps.ReexportHeaderLibHeaders = append(deps.ReexportHeaderLibHeaders, linker.Properties.Export_header_lib_headers...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700235 deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, linker.Properties.Export_static_lib_headers...)
236 deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, linker.Properties.Export_shared_lib_headers...)
Dan Willemsenb3454ab2016-09-28 17:34:58 -0700237 deps.ReexportGeneratedHeaders = append(deps.ReexportGeneratedHeaders, linker.Properties.Export_generated_headers...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700238
Colin Cross06210312020-06-10 15:47:34 -0700239 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Exclude_shared_libs)
240 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Exclude_static_libs)
Christopher Ferrisc71193a2019-12-16 09:55:10 -0800241 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Exclude_static_libs)
242
Colin Cross86803cf2018-02-15 14:12:26 -0800243 if Bool(linker.Properties.Use_version_lib) {
244 deps.WholeStaticLibs = append(deps.WholeStaticLibs, "libbuildversion")
245 }
246
Justin Yun63e9ec72020-10-29 16:49:43 +0900247 // TODO(b/150902910): product variant must use Target.Product
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700248 if ctx.useVndk() {
Martin Stjernholmef449fe2018-11-06 16:12:13 +0000249 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Target.Vendor.Shared_libs...)
Jiyong Park74472282017-08-10 00:48:06 +0900250 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Vendor.Exclude_shared_libs)
251 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Vendor.Exclude_shared_libs)
Chong Zhanged5c1462020-02-20 15:04:35 -0800252 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Target.Vendor.Static_libs...)
Jiyong Park52d25bd2017-10-13 09:17:01 +0900253 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Vendor.Exclude_static_libs)
Jiwen 'Steve' Cai8dbc6532018-08-08 10:58:34 -0700254 deps.HeaderLibs = removeListFromList(deps.HeaderLibs, linker.Properties.Target.Vendor.Exclude_header_libs)
Jiyong Park52d25bd2017-10-13 09:17:01 +0900255 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Vendor.Exclude_static_libs)
256 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Vendor.Exclude_static_libs)
Logan Chien43d34c32017-12-20 01:17:32 +0800257 deps.RuntimeLibs = removeListFromList(deps.RuntimeLibs, linker.Properties.Target.Vendor.Exclude_runtime_libs)
Jiyong Park74472282017-08-10 00:48:06 +0900258 }
259
Jiyong Parkf9332f12018-02-01 00:54:12 +0900260 if ctx.inRecovery() {
Martin Stjernholmef449fe2018-11-06 16:12:13 +0000261 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Target.Recovery.Shared_libs...)
Jiyong Parkf9332f12018-02-01 00:54:12 +0900262 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Recovery.Exclude_shared_libs)
263 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Recovery.Exclude_shared_libs)
Yifan Honge7efbc82019-08-01 13:45:42 -0700264 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Target.Recovery.Static_libs...)
Jiyong Parkf9332f12018-02-01 00:54:12 +0900265 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Recovery.Exclude_static_libs)
Hridya Valsaraju280febf2018-08-10 09:08:13 -0700266 deps.HeaderLibs = removeListFromList(deps.HeaderLibs, linker.Properties.Target.Recovery.Exclude_header_libs)
267 deps.ReexportHeaderLibHeaders = removeListFromList(deps.ReexportHeaderLibHeaders, linker.Properties.Target.Recovery.Exclude_header_libs)
Jiyong Parkf9332f12018-02-01 00:54:12 +0900268 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Recovery.Exclude_static_libs)
269 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Recovery.Exclude_static_libs)
270 }
271
Yifan Hongcf4832c2020-01-21 17:04:13 -0800272 if ctx.inRamdisk() {
Yifan Hongd2057fb2020-10-27 15:03:34 -0700273 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Ramdisk.Exclude_shared_libs)
274 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Ramdisk.Exclude_shared_libs)
275 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Target.Ramdisk.Static_libs...)
276 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Ramdisk.Exclude_static_libs)
277 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Ramdisk.Exclude_static_libs)
278 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Ramdisk.Exclude_static_libs)
Yifan Hongcf4832c2020-01-21 17:04:13 -0800279 }
280
Yifan Hong6da33c22020-10-27 15:01:21 -0700281 if ctx.inVendorRamdisk() {
282 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
283 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
284 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
285 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
286 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
287 }
288
Colin Crossc511bc52020-04-07 16:50:32 +0000289 if !ctx.useSdk() {
290 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Target.Platform.Shared_libs...)
291 }
292
Yi Kong6fb831b2018-10-05 19:46:01 +0000293 if ctx.toolchain().Bionic() {
Peter Collingbournee5ba2862019-12-10 18:37:45 -0800294 // libclang_rt.builtins and libatomic have to be last on the command line
Yi Kong2c188be2018-10-10 13:43:02 -0700295 if !Bool(linker.Properties.No_libcrt) {
Yi Kong19408a32019-02-12 09:41:18 -0800296 deps.LateStaticLibs = append(deps.LateStaticLibs, config.BuiltinsRuntimeLibrary(ctx.toolchain()))
Yi Kongacee27c2019-03-29 20:05:14 -0700297 deps.LateStaticLibs = append(deps.LateStaticLibs, "libatomic")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700298 }
299
Yo Chiang219968c2020-09-22 18:45:04 +0800300 deps.SystemSharedLibs = linker.Properties.System_shared_libs
301 if deps.SystemSharedLibs == nil {
Logan Chien14bd0db2019-01-15 21:12:43 +0800302 // Provide a default system_shared_libs if it is unspecified. Note: If an
303 // empty list [] is specified, it implies that the module declines the
304 // default system_shared_libs.
Yo Chiang219968c2020-09-22 18:45:04 +0800305 deps.SystemSharedLibs = []string{"libc", "libm", "libdl"}
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800306 }
307
308 if inList("libdl", deps.SharedLibs) {
309 // If system_shared_libs has libc but not libdl, make sure shared_libs does not
310 // have libdl to avoid loading libdl before libc.
Yo Chiang219968c2020-09-22 18:45:04 +0800311 if inList("libc", deps.SystemSharedLibs) {
312 if !inList("libdl", deps.SystemSharedLibs) {
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800313 ctx.PropertyErrorf("shared_libs",
314 "libdl must be in system_shared_libs, not shared_libs")
315 }
316 _, deps.SharedLibs = removeFromList("libdl", deps.SharedLibs)
317 }
318 }
319
320 // If libc and libdl are both in system_shared_libs make sure libdl comes after libc
321 // to avoid loading libdl before libc.
Yo Chiang219968c2020-09-22 18:45:04 +0800322 if inList("libdl", deps.SystemSharedLibs) && inList("libc", deps.SystemSharedLibs) &&
323 indexList("libdl", deps.SystemSharedLibs) < indexList("libc", deps.SystemSharedLibs) {
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800324 ctx.PropertyErrorf("system_shared_libs", "libdl must be after libc")
325 }
326
Yo Chiang219968c2020-09-22 18:45:04 +0800327 deps.LateSharedLibs = append(deps.LateSharedLibs, deps.SystemSharedLibs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700328 }
329
Doug Hornc32c6b02019-01-17 14:44:05 -0800330 if ctx.Fuchsia() {
331 if ctx.ModuleName() != "libbioniccompat" &&
332 ctx.ModuleName() != "libcompiler_rt-extras" &&
333 ctx.ModuleName() != "libcompiler_rt" {
334 deps.StaticLibs = append(deps.StaticLibs, "libbioniccompat")
335 }
336 if ctx.ModuleName() != "libcompiler_rt" && ctx.ModuleName() != "libcompiler_rt-extras" {
337 deps.LateStaticLibs = append(deps.LateStaticLibs, "libcompiler_rt")
338 }
339
340 }
341
Colin Cross3edeee12017-04-04 12:59:48 -0700342 if ctx.Windows() {
Josh Gao7bd4c5c2017-02-23 17:52:24 -0800343 deps.LateStaticLibs = append(deps.LateStaticLibs, "libwinpthread")
344 }
345
Colin Cross4d9c2d12016-07-29 12:48:20 -0700346 return deps
347}
348
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700349func (linker *baseLinker) useClangLld(ctx ModuleContext) bool {
Chih-Hung Hsiehe5ac6092018-04-24 16:00:01 -0700350 // Clang lld is not ready for for Darwin host executables yet.
351 // See https://lld.llvm.org/AtomLLD.html for status of lld for Mach-O.
352 if ctx.Darwin() {
353 return false
354 }
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700355 if linker.Properties.Use_clang_lld != nil {
356 return Bool(linker.Properties.Use_clang_lld)
357 }
Dan Willemsenfa2aee12018-10-21 19:47:01 -0700358 return true
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700359}
360
Zhizhou Yang9100b1d2018-11-30 14:00:04 -0800361// Check whether the SDK version is not older than the specific one
362func CheckSdkVersionAtLeast(ctx ModuleContext, SdkVersion int) bool {
363 if ctx.sdkVersion() == "current" {
364 return true
365 }
366 parsedSdkVersion, err := strconv.Atoi(ctx.sdkVersion())
367 if err != nil {
368 ctx.PropertyErrorf("sdk_version",
369 "Invalid sdk_version value (must be int or current): %q",
370 ctx.sdkVersion())
371 }
372 if parsedSdkVersion < SdkVersion {
373 return false
374 }
375 return true
376}
377
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700378// ModuleContext extends BaseModuleContext
379// BaseModuleContext should know if LLD is used?
Colin Cross42742b82016-08-01 13:20:05 -0700380func (linker *baseLinker) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700381 toolchain := ctx.toolchain()
382
Colin Cross324a4572017-11-02 23:09:41 -0700383 hod := "Host"
384 if ctx.Os().Class == android.Device {
385 hod = "Device"
386 }
387
Dan Willemsen8536d6b2018-10-07 20:54:34 -0700388 if linker.useClangLld(ctx) {
Colin Cross4af21ed2019-11-04 09:37:55 -0800389 flags.Global.LdFlags = append(flags.Global.LdFlags, fmt.Sprintf("${config.%sGlobalLldflags}", hod))
Colin Crossc17727d2018-10-24 12:42:09 -0700390 if !BoolDefault(linker.Properties.Pack_relocations, true) {
Colin Cross4af21ed2019-11-04 09:37:55 -0800391 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--pack-dyn-relocs=none")
Zhizhou Yang9100b1d2018-11-30 14:00:04 -0800392 } else if ctx.Device() {
393 // The SHT_RELR relocations is only supported by API level >= 28.
394 // Do not turn this on if older version NDK is used.
395 if !ctx.useSdk() || CheckSdkVersionAtLeast(ctx, 28) {
Colin Cross4af21ed2019-11-04 09:37:55 -0800396 flags.Global.LdFlags = append(flags.Global.LdFlags,
397 "-Wl,--pack-dyn-relocs=android+relr",
398 "-Wl,--use-android-relr-tags")
Elliott Hughesda909fe2020-01-28 13:08:40 -0800399 } else if CheckSdkVersionAtLeast(ctx, 23) {
400 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--pack-dyn-relocs=android")
Zhizhou Yang9100b1d2018-11-30 14:00:04 -0800401 }
Chih-Hung Hsieh86814712018-05-23 18:30:46 -0700402 }
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700403 } else {
Colin Cross4af21ed2019-11-04 09:37:55 -0800404 flags.Global.LdFlags = append(flags.Global.LdFlags, fmt.Sprintf("${config.%sGlobalLdflags}", hod))
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700405 }
Colin Cross87dd9632017-11-03 13:31:05 -0700406 if Bool(linker.Properties.Allow_undefined_symbols) {
407 if ctx.Darwin() {
408 // darwin defaults to treating undefined symbols as errors
Colin Cross4af21ed2019-11-04 09:37:55 -0800409 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,-undefined,dynamic_lookup")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700410 }
Josh Gao75a50a22019-06-07 17:58:59 -0700411 } else if !ctx.Darwin() && !ctx.Windows() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800412 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--no-undefined")
Colin Cross87dd9632017-11-03 13:31:05 -0700413 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700414
Dan Willemsen8536d6b2018-10-07 20:54:34 -0700415 if linker.useClangLld(ctx) {
Colin Cross4af21ed2019-11-04 09:37:55 -0800416 flags.Global.LdFlags = append(flags.Global.LdFlags, toolchain.ClangLldflags())
Colin Cross87dd9632017-11-03 13:31:05 -0700417 } else {
Colin Cross4af21ed2019-11-04 09:37:55 -0800418 flags.Global.LdFlags = append(flags.Global.LdFlags, toolchain.ClangLdflags())
Colin Cross87dd9632017-11-03 13:31:05 -0700419 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700420
Doug Hornc32c6b02019-01-17 14:44:05 -0800421 if !ctx.toolchain().Bionic() && !ctx.Fuchsia() {
Colin Cross87dd9632017-11-03 13:31:05 -0700422 CheckBadHostLdlibs(ctx, "host_ldlibs", linker.Properties.Host_ldlibs)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700423
Colin Cross4af21ed2019-11-04 09:37:55 -0800424 flags.Local.LdFlags = append(flags.Local.LdFlags, linker.Properties.Host_ldlibs...)
Colin Crossc5fdbb82017-09-08 12:45:18 -0700425
Colin Cross87dd9632017-11-03 13:31:05 -0700426 if !ctx.Windows() {
427 // Add -ldl, -lpthread, -lm and -lrt to host builds to match the default behavior of device
428 // builds
Colin Cross4af21ed2019-11-04 09:37:55 -0800429 flags.Global.LdFlags = append(flags.Global.LdFlags,
Colin Cross87dd9632017-11-03 13:31:05 -0700430 "-ldl",
431 "-lpthread",
432 "-lm",
433 )
434 if !ctx.Darwin() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800435 flags.Global.LdFlags = append(flags.Global.LdFlags, "-lrt")
Colin Crossc5fdbb82017-09-08 12:45:18 -0700436 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700437 }
438 }
439
Doug Hornc32c6b02019-01-17 14:44:05 -0800440 if ctx.Fuchsia() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800441 flags.Global.LdFlags = append(flags.Global.LdFlags, "-lfdio", "-lzircon")
Doug Hornc32c6b02019-01-17 14:44:05 -0800442 }
443
Tom Cherrye4802322019-08-06 10:03:25 -0700444 if ctx.toolchain().LibclangRuntimeLibraryArch() != "" {
Colin Cross4af21ed2019-11-04 09:37:55 -0800445 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--exclude-libs="+config.BuiltinsRuntimeLibrary(ctx.toolchain())+".a")
Tom Cherrye4802322019-08-06 10:03:25 -0700446 }
447
Colin Cross4d9c2d12016-07-29 12:48:20 -0700448 CheckBadLinkerFlags(ctx, "ldflags", linker.Properties.Ldflags)
449
Colin Cross4af21ed2019-11-04 09:37:55 -0800450 flags.Local.LdFlags = append(flags.Local.LdFlags, proptools.NinjaAndShellEscapeList(linker.Properties.Ldflags)...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700451
Josh Gao75a50a22019-06-07 17:58:59 -0700452 if ctx.Host() && !ctx.Windows() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700453 rpath_prefix := `\$$ORIGIN/`
454 if ctx.Darwin() {
455 rpath_prefix = "@loader_path/"
456 }
457
Dan Willemsen6f91fbf2017-09-18 22:45:15 -0700458 if !ctx.static() {
459 for _, rpath := range linker.dynamicProperties.RunPaths {
Colin Cross4af21ed2019-11-04 09:37:55 -0800460 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,-rpath,"+rpath_prefix+rpath)
Dan Willemsen6f91fbf2017-09-18 22:45:15 -0700461 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700462 }
463 }
464
Elliott Hughesda3a0712020-03-06 16:55:28 -0800465 if ctx.useSdk() {
Colin Cross6774e282017-08-11 13:23:57 -0700466 // The bionic linker now has support gnu style hashes (which are much faster!), but shipping
467 // to older devices requires the old style hash. Fortunately, we can build with both and
468 // it'll work anywhere.
Colin Cross4af21ed2019-11-04 09:37:55 -0800469 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--hash-style=both")
Colin Cross6774e282017-08-11 13:23:57 -0700470 }
471
Colin Cross4af21ed2019-11-04 09:37:55 -0800472 flags.Global.LdFlags = append(flags.Global.LdFlags, toolchain.ToolchainClangLdflags())
Colin Cross4d9c2d12016-07-29 12:48:20 -0700473
Colin Cross18c0c5a2016-12-01 14:45:23 -0800474 if Bool(linker.Properties.Group_static_libs) {
475 flags.GroupStaticLibs = true
476 }
477
Jiyong Park7ed9de32018-10-15 22:25:07 +0900478 // Version_script is not needed when linking stubs lib where the version
479 // script is created from the symbol map file.
480 if !linker.dynamicProperties.BuildStubs {
481 versionScript := ctx.ExpandOptionalSource(
Colin Crossc17727d2018-10-24 12:42:09 -0700482 linker.Properties.Version_script, "version_script")
Dan Albert61f32122018-07-26 14:00:24 -0700483
Justin Yun63e9ec72020-10-29 16:49:43 +0900484 // TODO(b/150902910): product variant must use Target.Product
Colin Crossc17727d2018-10-24 12:42:09 -0700485 if ctx.useVndk() && linker.Properties.Target.Vendor.Version_script != nil {
Jiyong Park7ed9de32018-10-15 22:25:07 +0900486 versionScript = ctx.ExpandOptionalSource(
Colin Crossc17727d2018-10-24 12:42:09 -0700487 linker.Properties.Target.Vendor.Version_script,
Jiyong Park7ed9de32018-10-15 22:25:07 +0900488 "target.vendor.version_script")
489 }
Dan Albert61f32122018-07-26 14:00:24 -0700490
Jiyong Park7ed9de32018-10-15 22:25:07 +0900491 if versionScript.Valid() {
492 if ctx.Darwin() {
493 ctx.PropertyErrorf("version_script", "Not supported on Darwin")
494 } else {
Colin Cross4af21ed2019-11-04 09:37:55 -0800495 flags.Local.LdFlags = append(flags.Local.LdFlags,
Jiyong Park7ed9de32018-10-15 22:25:07 +0900496 "-Wl,--version-script,"+versionScript.String())
497 flags.LdFlagsDeps = append(flags.LdFlagsDeps, versionScript.Path())
498
499 if linker.sanitize.isSanitizerEnabled(cfi) {
500 cfiExportsMap := android.PathForSource(ctx, cfiExportsMapPath)
Colin Cross4af21ed2019-11-04 09:37:55 -0800501 flags.Local.LdFlags = append(flags.Local.LdFlags,
Jiyong Park7ed9de32018-10-15 22:25:07 +0900502 "-Wl,--version-script,"+cfiExportsMap.String())
503 flags.LdFlagsDeps = append(flags.LdFlagsDeps, cfiExportsMap)
504 }
Dan Albert61f32122018-07-26 14:00:24 -0700505 }
506 }
507 }
508
Colin Cross4d9c2d12016-07-29 12:48:20 -0700509 return flags
510}
511
Colin Crossb916a382016-07-29 17:28:03 -0700512func (linker *baseLinker) link(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700513 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Crossb916a382016-07-29 17:28:03 -0700514 panic(fmt.Errorf("baseLinker doesn't know how to link"))
Colin Cross4d9c2d12016-07-29 12:48:20 -0700515}
Colin Cross86803cf2018-02-15 14:12:26 -0800516
Paul Duffin13f02712020-03-06 12:30:43 +0000517func (linker *baseLinker) linkerSpecifiedDeps(specifiedDeps specifiedDeps) specifiedDeps {
518 specifiedDeps.sharedLibs = append(specifiedDeps.sharedLibs, linker.Properties.Shared_libs...)
Martin Stjernholm10566a02020-03-24 01:19:52 +0000519
520 // Must distinguish nil and [] in system_shared_libs - ensure that [] in
521 // either input list doesn't come out as nil.
522 if specifiedDeps.systemSharedLibs == nil {
523 specifiedDeps.systemSharedLibs = linker.Properties.System_shared_libs
524 } else {
525 specifiedDeps.systemSharedLibs = append(specifiedDeps.systemSharedLibs, linker.Properties.System_shared_libs...)
526 }
527
Paul Duffin13f02712020-03-06 12:30:43 +0000528 return specifiedDeps
529}
530
Colin Cross86803cf2018-02-15 14:12:26 -0800531// Injecting version symbols
532// Some host modules want a version number, but we don't want to rebuild it every time. Optionally add a step
533// after linking that injects a constant placeholder with the current version number.
534
535func init() {
536 pctx.HostBinToolVariable("symbolInjectCmd", "symbol_inject")
537}
538
539var injectVersionSymbol = pctx.AndroidStaticRule("injectVersionSymbol",
540 blueprint.RuleParams{
541 Command: "$symbolInjectCmd -i $in -o $out -s soong_build_number " +
Colin Cross2a2e0db2020-02-21 16:55:46 -0800542 "-from 'SOONG BUILD NUMBER PLACEHOLDER' -v $$(cat $buildNumberFile)",
Colin Cross86803cf2018-02-15 14:12:26 -0800543 CommandDeps: []string{"$symbolInjectCmd"},
544 },
Colin Cross2a2e0db2020-02-21 16:55:46 -0800545 "buildNumberFile")
Colin Cross86803cf2018-02-15 14:12:26 -0800546
547func (linker *baseLinker) injectVersionSymbol(ctx ModuleContext, in android.Path, out android.WritablePath) {
Colin Cross2a2e0db2020-02-21 16:55:46 -0800548 buildNumberFile := ctx.Config().BuildNumberFile(ctx)
Colin Cross86803cf2018-02-15 14:12:26 -0800549 ctx.Build(pctx, android.BuildParams{
550 Rule: injectVersionSymbol,
551 Description: "inject version symbol",
552 Input: in,
553 Output: out,
Colin Cross2a2e0db2020-02-21 16:55:46 -0800554 OrderOnly: android.Paths{buildNumberFile},
Colin Cross86803cf2018-02-15 14:12:26 -0800555 Args: map[string]string{
Colin Cross2a2e0db2020-02-21 16:55:46 -0800556 "buildNumberFile": buildNumberFile.String(),
Colin Cross86803cf2018-02-15 14:12:26 -0800557 },
558 })
559}
Vic Yang6cd1be82019-06-24 16:08:48 -0700560
561// Rule to generate .bss symbol ordering file.
562
563var (
564 _ = pctx.SourcePathVariable("genSortedBssSymbolsPath", "build/soong/scripts/gen_sorted_bss_symbols.sh")
565 gen_sorted_bss_symbols = pctx.AndroidStaticRule("gen_sorted_bss_symbols",
566 blueprint.RuleParams{
567 Command: "CROSS_COMPILE=$crossCompile $genSortedBssSymbolsPath ${in} ${out}",
Dan Willemsen724ab5d2019-09-19 10:50:18 -0700568 CommandDeps: []string{"$genSortedBssSymbolsPath", "${crossCompile}nm"},
Vic Yang6cd1be82019-06-24 16:08:48 -0700569 },
570 "crossCompile")
571)
572
573func (linker *baseLinker) sortBssSymbolsBySize(ctx ModuleContext, in android.Path, symbolOrderingFile android.ModuleOutPath, flags builderFlags) string {
574 crossCompile := gccCmd(flags.toolchain, "")
575 ctx.Build(pctx, android.BuildParams{
576 Rule: gen_sorted_bss_symbols,
577 Description: "generate bss symbol order " + symbolOrderingFile.Base(),
578 Output: symbolOrderingFile,
579 Input: in,
580 Args: map[string]string{
581 "crossCompile": crossCompile,
582 },
583 })
584 return "-Wl,--symbol-ordering-file," + symbolOrderingFile.String()
585}