blob: 7f13e288de8753948dea967cb8e4f159c1a7f455 [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 {
99 Vendor struct {
Martin Stjernholmef449fe2018-11-06 16:12:13 +0000100 // list of shared libs that only should be used to build the vendor
101 // variant of the C/C++ module.
102 Shared_libs []string
103
Chong Zhanged5c1462020-02-20 15:04:35 -0800104 // list of static libs that only should be used to build the vendor
105 // variant of the C/C++ module.
106 Static_libs []string
107
Logan Chien43d34c32017-12-20 01:17:32 +0800108 // list of shared libs that should not be used to build the vendor variant
109 // 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
Logan Chien43d34c32017-12-20 01:17:32 +0800112 // list of static libs that should not be used to build the vendor variant
113 // 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
Jiwen 'Steve' Cai8dbc6532018-08-08 10:58:34 -0700116 // list of header libs that should not be used to build the vendor variant
117 // of the C/C++ module.
118 Exclude_header_libs []string
119
Logan Chien43d34c32017-12-20 01:17:32 +0800120 // list of runtime libs that should not be installed along with the vendor
121 // variant of the C/C++ module.
122 Exclude_runtime_libs []string
Colin Crossc17727d2018-10-24 12:42:09 -0700123
124 // version script for this vendor variant
125 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
Colin Cross42742b82016-08-01 13:20:05 -0700214func (linker *baseLinker) linkerInit(ctx BaseModuleContext) {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700215 if ctx.toolchain().Is64Bit() {
Colin Crossb916a382016-07-29 17:28:03 -0700216 linker.dynamicProperties.RunPaths = append(linker.dynamicProperties.RunPaths, "../lib64", "lib64")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700217 } else {
Colin Crossb916a382016-07-29 17:28:03 -0700218 linker.dynamicProperties.RunPaths = append(linker.dynamicProperties.RunPaths, "../lib", "lib")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700219 }
220}
221
Colin Cross42742b82016-08-01 13:20:05 -0700222func (linker *baseLinker) linkerProps() []interface{} {
Colin Crossc17727d2018-10-24 12:42:09 -0700223 return []interface{}{&linker.Properties, &linker.dynamicProperties}
Colin Cross4d9c2d12016-07-29 12:48:20 -0700224}
225
Dan Albert61f32122018-07-26 14:00:24 -0700226func (linker *baseLinker) linkerDeps(ctx DepsContext, deps Deps) Deps {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700227 deps.WholeStaticLibs = append(deps.WholeStaticLibs, linker.Properties.Whole_static_libs...)
Colin Cross5950f382016-12-13 12:50:57 -0800228 deps.HeaderLibs = append(deps.HeaderLibs, linker.Properties.Header_libs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700229 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Static_libs...)
230 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Shared_libs...)
Logan Chien43d34c32017-12-20 01:17:32 +0800231 deps.RuntimeLibs = append(deps.RuntimeLibs, linker.Properties.Runtime_libs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700232
Colin Cross5950f382016-12-13 12:50:57 -0800233 deps.ReexportHeaderLibHeaders = append(deps.ReexportHeaderLibHeaders, linker.Properties.Export_header_lib_headers...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700234 deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, linker.Properties.Export_static_lib_headers...)
235 deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, linker.Properties.Export_shared_lib_headers...)
Dan Willemsenb3454ab2016-09-28 17:34:58 -0700236 deps.ReexportGeneratedHeaders = append(deps.ReexportGeneratedHeaders, linker.Properties.Export_generated_headers...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700237
Colin Cross06210312020-06-10 15:47:34 -0700238 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Exclude_shared_libs)
239 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Exclude_static_libs)
Christopher Ferrisc71193a2019-12-16 09:55:10 -0800240 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Exclude_static_libs)
241
Colin Cross86803cf2018-02-15 14:12:26 -0800242 if Bool(linker.Properties.Use_version_lib) {
243 deps.WholeStaticLibs = append(deps.WholeStaticLibs, "libbuildversion")
244 }
245
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700246 if ctx.useVndk() {
Martin Stjernholmef449fe2018-11-06 16:12:13 +0000247 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Target.Vendor.Shared_libs...)
Jiyong Park74472282017-08-10 00:48:06 +0900248 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Vendor.Exclude_shared_libs)
249 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Vendor.Exclude_shared_libs)
Chong Zhanged5c1462020-02-20 15:04:35 -0800250 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Target.Vendor.Static_libs...)
Jiyong Park52d25bd2017-10-13 09:17:01 +0900251 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Vendor.Exclude_static_libs)
Jiwen 'Steve' Cai8dbc6532018-08-08 10:58:34 -0700252 deps.HeaderLibs = removeListFromList(deps.HeaderLibs, linker.Properties.Target.Vendor.Exclude_header_libs)
Jiyong Park52d25bd2017-10-13 09:17:01 +0900253 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Vendor.Exclude_static_libs)
254 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Vendor.Exclude_static_libs)
Logan Chien43d34c32017-12-20 01:17:32 +0800255 deps.RuntimeLibs = removeListFromList(deps.RuntimeLibs, linker.Properties.Target.Vendor.Exclude_runtime_libs)
Jiyong Park74472282017-08-10 00:48:06 +0900256 }
257
Jiyong Parkf9332f12018-02-01 00:54:12 +0900258 if ctx.inRecovery() {
Martin Stjernholmef449fe2018-11-06 16:12:13 +0000259 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Target.Recovery.Shared_libs...)
Jiyong Parkf9332f12018-02-01 00:54:12 +0900260 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Recovery.Exclude_shared_libs)
261 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Recovery.Exclude_shared_libs)
Yifan Honge7efbc82019-08-01 13:45:42 -0700262 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Target.Recovery.Static_libs...)
Jiyong Parkf9332f12018-02-01 00:54:12 +0900263 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Recovery.Exclude_static_libs)
Hridya Valsaraju280febf2018-08-10 09:08:13 -0700264 deps.HeaderLibs = removeListFromList(deps.HeaderLibs, linker.Properties.Target.Recovery.Exclude_header_libs)
265 deps.ReexportHeaderLibHeaders = removeListFromList(deps.ReexportHeaderLibHeaders, linker.Properties.Target.Recovery.Exclude_header_libs)
Jiyong Parkf9332f12018-02-01 00:54:12 +0900266 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Recovery.Exclude_static_libs)
267 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Recovery.Exclude_static_libs)
268 }
269
Yifan Hongcf4832c2020-01-21 17:04:13 -0800270 if ctx.inRamdisk() {
Yifan Hongd2057fb2020-10-27 15:03:34 -0700271 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Ramdisk.Exclude_shared_libs)
272 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Ramdisk.Exclude_shared_libs)
273 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Target.Ramdisk.Static_libs...)
274 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Ramdisk.Exclude_static_libs)
275 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Ramdisk.Exclude_static_libs)
276 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Ramdisk.Exclude_static_libs)
Yifan Hongcf4832c2020-01-21 17:04:13 -0800277 }
278
Yifan Hong6da33c22020-10-27 15:01:21 -0700279 if ctx.inVendorRamdisk() {
280 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
281 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
282 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
283 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
284 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
285 }
286
Colin Crossc511bc52020-04-07 16:50:32 +0000287 if !ctx.useSdk() {
288 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Target.Platform.Shared_libs...)
289 }
290
Yi Kong6fb831b2018-10-05 19:46:01 +0000291 if ctx.toolchain().Bionic() {
Peter Collingbournee5ba2862019-12-10 18:37:45 -0800292 // libclang_rt.builtins and libatomic have to be last on the command line
Yi Kong2c188be2018-10-10 13:43:02 -0700293 if !Bool(linker.Properties.No_libcrt) {
Yi Kong19408a32019-02-12 09:41:18 -0800294 deps.LateStaticLibs = append(deps.LateStaticLibs, config.BuiltinsRuntimeLibrary(ctx.toolchain()))
Yi Kongacee27c2019-03-29 20:05:14 -0700295 deps.LateStaticLibs = append(deps.LateStaticLibs, "libatomic")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700296 }
297
Yo Chiang219968c2020-09-22 18:45:04 +0800298 deps.SystemSharedLibs = linker.Properties.System_shared_libs
299 if deps.SystemSharedLibs == nil {
Logan Chien14bd0db2019-01-15 21:12:43 +0800300 // Provide a default system_shared_libs if it is unspecified. Note: If an
301 // empty list [] is specified, it implies that the module declines the
302 // default system_shared_libs.
Yo Chiang219968c2020-09-22 18:45:04 +0800303 deps.SystemSharedLibs = []string{"libc", "libm", "libdl"}
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800304 }
305
306 if inList("libdl", deps.SharedLibs) {
307 // If system_shared_libs has libc but not libdl, make sure shared_libs does not
308 // have libdl to avoid loading libdl before libc.
Yo Chiang219968c2020-09-22 18:45:04 +0800309 if inList("libc", deps.SystemSharedLibs) {
310 if !inList("libdl", deps.SystemSharedLibs) {
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800311 ctx.PropertyErrorf("shared_libs",
312 "libdl must be in system_shared_libs, not shared_libs")
313 }
314 _, deps.SharedLibs = removeFromList("libdl", deps.SharedLibs)
315 }
316 }
317
318 // If libc and libdl are both in system_shared_libs make sure libdl comes after libc
319 // to avoid loading libdl before libc.
Yo Chiang219968c2020-09-22 18:45:04 +0800320 if inList("libdl", deps.SystemSharedLibs) && inList("libc", deps.SystemSharedLibs) &&
321 indexList("libdl", deps.SystemSharedLibs) < indexList("libc", deps.SystemSharedLibs) {
Dan Willemsen3a26eef2018-12-03 15:25:46 -0800322 ctx.PropertyErrorf("system_shared_libs", "libdl must be after libc")
323 }
324
Yo Chiang219968c2020-09-22 18:45:04 +0800325 deps.LateSharedLibs = append(deps.LateSharedLibs, deps.SystemSharedLibs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700326 }
327
Doug Hornc32c6b02019-01-17 14:44:05 -0800328 if ctx.Fuchsia() {
329 if ctx.ModuleName() != "libbioniccompat" &&
330 ctx.ModuleName() != "libcompiler_rt-extras" &&
331 ctx.ModuleName() != "libcompiler_rt" {
332 deps.StaticLibs = append(deps.StaticLibs, "libbioniccompat")
333 }
334 if ctx.ModuleName() != "libcompiler_rt" && ctx.ModuleName() != "libcompiler_rt-extras" {
335 deps.LateStaticLibs = append(deps.LateStaticLibs, "libcompiler_rt")
336 }
337
338 }
339
Colin Cross3edeee12017-04-04 12:59:48 -0700340 if ctx.Windows() {
Josh Gao7bd4c5c2017-02-23 17:52:24 -0800341 deps.LateStaticLibs = append(deps.LateStaticLibs, "libwinpthread")
342 }
343
Colin Cross4d9c2d12016-07-29 12:48:20 -0700344 return deps
345}
346
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700347func (linker *baseLinker) useClangLld(ctx ModuleContext) bool {
Chih-Hung Hsiehe5ac6092018-04-24 16:00:01 -0700348 // Clang lld is not ready for for Darwin host executables yet.
349 // See https://lld.llvm.org/AtomLLD.html for status of lld for Mach-O.
350 if ctx.Darwin() {
351 return false
352 }
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700353 if linker.Properties.Use_clang_lld != nil {
354 return Bool(linker.Properties.Use_clang_lld)
355 }
Dan Willemsenfa2aee12018-10-21 19:47:01 -0700356 return true
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700357}
358
Zhizhou Yang9100b1d2018-11-30 14:00:04 -0800359// Check whether the SDK version is not older than the specific one
360func CheckSdkVersionAtLeast(ctx ModuleContext, SdkVersion int) bool {
361 if ctx.sdkVersion() == "current" {
362 return true
363 }
364 parsedSdkVersion, err := strconv.Atoi(ctx.sdkVersion())
365 if err != nil {
366 ctx.PropertyErrorf("sdk_version",
367 "Invalid sdk_version value (must be int or current): %q",
368 ctx.sdkVersion())
369 }
370 if parsedSdkVersion < SdkVersion {
371 return false
372 }
373 return true
374}
375
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700376// ModuleContext extends BaseModuleContext
377// BaseModuleContext should know if LLD is used?
Colin Cross42742b82016-08-01 13:20:05 -0700378func (linker *baseLinker) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700379 toolchain := ctx.toolchain()
380
Colin Cross324a4572017-11-02 23:09:41 -0700381 hod := "Host"
382 if ctx.Os().Class == android.Device {
383 hod = "Device"
384 }
385
Dan Willemsen8536d6b2018-10-07 20:54:34 -0700386 if linker.useClangLld(ctx) {
Colin Cross4af21ed2019-11-04 09:37:55 -0800387 flags.Global.LdFlags = append(flags.Global.LdFlags, fmt.Sprintf("${config.%sGlobalLldflags}", hod))
Colin Crossc17727d2018-10-24 12:42:09 -0700388 if !BoolDefault(linker.Properties.Pack_relocations, true) {
Colin Cross4af21ed2019-11-04 09:37:55 -0800389 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--pack-dyn-relocs=none")
Zhizhou Yang9100b1d2018-11-30 14:00:04 -0800390 } else if ctx.Device() {
391 // The SHT_RELR relocations is only supported by API level >= 28.
392 // Do not turn this on if older version NDK is used.
393 if !ctx.useSdk() || CheckSdkVersionAtLeast(ctx, 28) {
Colin Cross4af21ed2019-11-04 09:37:55 -0800394 flags.Global.LdFlags = append(flags.Global.LdFlags,
395 "-Wl,--pack-dyn-relocs=android+relr",
396 "-Wl,--use-android-relr-tags")
Elliott Hughesda909fe2020-01-28 13:08:40 -0800397 } else if CheckSdkVersionAtLeast(ctx, 23) {
398 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--pack-dyn-relocs=android")
Zhizhou Yang9100b1d2018-11-30 14:00:04 -0800399 }
Chih-Hung Hsieh86814712018-05-23 18:30:46 -0700400 }
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700401 } else {
Colin Cross4af21ed2019-11-04 09:37:55 -0800402 flags.Global.LdFlags = append(flags.Global.LdFlags, fmt.Sprintf("${config.%sGlobalLdflags}", hod))
Chih-Hung Hsieh02b4da52018-04-03 11:33:34 -0700403 }
Colin Cross87dd9632017-11-03 13:31:05 -0700404 if Bool(linker.Properties.Allow_undefined_symbols) {
405 if ctx.Darwin() {
406 // darwin defaults to treating undefined symbols as errors
Colin Cross4af21ed2019-11-04 09:37:55 -0800407 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,-undefined,dynamic_lookup")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700408 }
Josh Gao75a50a22019-06-07 17:58:59 -0700409 } else if !ctx.Darwin() && !ctx.Windows() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800410 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--no-undefined")
Colin Cross87dd9632017-11-03 13:31:05 -0700411 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700412
Dan Willemsen8536d6b2018-10-07 20:54:34 -0700413 if linker.useClangLld(ctx) {
Colin Cross4af21ed2019-11-04 09:37:55 -0800414 flags.Global.LdFlags = append(flags.Global.LdFlags, toolchain.ClangLldflags())
Colin Cross87dd9632017-11-03 13:31:05 -0700415 } else {
Colin Cross4af21ed2019-11-04 09:37:55 -0800416 flags.Global.LdFlags = append(flags.Global.LdFlags, toolchain.ClangLdflags())
Colin Cross87dd9632017-11-03 13:31:05 -0700417 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700418
Doug Hornc32c6b02019-01-17 14:44:05 -0800419 if !ctx.toolchain().Bionic() && !ctx.Fuchsia() {
Colin Cross87dd9632017-11-03 13:31:05 -0700420 CheckBadHostLdlibs(ctx, "host_ldlibs", linker.Properties.Host_ldlibs)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700421
Colin Cross4af21ed2019-11-04 09:37:55 -0800422 flags.Local.LdFlags = append(flags.Local.LdFlags, linker.Properties.Host_ldlibs...)
Colin Crossc5fdbb82017-09-08 12:45:18 -0700423
Colin Cross87dd9632017-11-03 13:31:05 -0700424 if !ctx.Windows() {
425 // Add -ldl, -lpthread, -lm and -lrt to host builds to match the default behavior of device
426 // builds
Colin Cross4af21ed2019-11-04 09:37:55 -0800427 flags.Global.LdFlags = append(flags.Global.LdFlags,
Colin Cross87dd9632017-11-03 13:31:05 -0700428 "-ldl",
429 "-lpthread",
430 "-lm",
431 )
432 if !ctx.Darwin() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800433 flags.Global.LdFlags = append(flags.Global.LdFlags, "-lrt")
Colin Crossc5fdbb82017-09-08 12:45:18 -0700434 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700435 }
436 }
437
Doug Hornc32c6b02019-01-17 14:44:05 -0800438 if ctx.Fuchsia() {
Colin Cross4af21ed2019-11-04 09:37:55 -0800439 flags.Global.LdFlags = append(flags.Global.LdFlags, "-lfdio", "-lzircon")
Doug Hornc32c6b02019-01-17 14:44:05 -0800440 }
441
Tom Cherrye4802322019-08-06 10:03:25 -0700442 if ctx.toolchain().LibclangRuntimeLibraryArch() != "" {
Colin Cross4af21ed2019-11-04 09:37:55 -0800443 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--exclude-libs="+config.BuiltinsRuntimeLibrary(ctx.toolchain())+".a")
Tom Cherrye4802322019-08-06 10:03:25 -0700444 }
445
Colin Cross4d9c2d12016-07-29 12:48:20 -0700446 CheckBadLinkerFlags(ctx, "ldflags", linker.Properties.Ldflags)
447
Colin Cross4af21ed2019-11-04 09:37:55 -0800448 flags.Local.LdFlags = append(flags.Local.LdFlags, proptools.NinjaAndShellEscapeList(linker.Properties.Ldflags)...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700449
Josh Gao75a50a22019-06-07 17:58:59 -0700450 if ctx.Host() && !ctx.Windows() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700451 rpath_prefix := `\$$ORIGIN/`
452 if ctx.Darwin() {
453 rpath_prefix = "@loader_path/"
454 }
455
Dan Willemsen6f91fbf2017-09-18 22:45:15 -0700456 if !ctx.static() {
457 for _, rpath := range linker.dynamicProperties.RunPaths {
Colin Cross4af21ed2019-11-04 09:37:55 -0800458 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,-rpath,"+rpath_prefix+rpath)
Dan Willemsen6f91fbf2017-09-18 22:45:15 -0700459 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700460 }
461 }
462
Elliott Hughesda3a0712020-03-06 16:55:28 -0800463 if ctx.useSdk() {
Colin Cross6774e282017-08-11 13:23:57 -0700464 // The bionic linker now has support gnu style hashes (which are much faster!), but shipping
465 // to older devices requires the old style hash. Fortunately, we can build with both and
466 // it'll work anywhere.
Colin Cross4af21ed2019-11-04 09:37:55 -0800467 flags.Global.LdFlags = append(flags.Global.LdFlags, "-Wl,--hash-style=both")
Colin Cross6774e282017-08-11 13:23:57 -0700468 }
469
Colin Cross4af21ed2019-11-04 09:37:55 -0800470 flags.Global.LdFlags = append(flags.Global.LdFlags, toolchain.ToolchainClangLdflags())
Colin Cross4d9c2d12016-07-29 12:48:20 -0700471
Colin Cross18c0c5a2016-12-01 14:45:23 -0800472 if Bool(linker.Properties.Group_static_libs) {
473 flags.GroupStaticLibs = true
474 }
475
Jiyong Park7ed9de32018-10-15 22:25:07 +0900476 // Version_script is not needed when linking stubs lib where the version
477 // script is created from the symbol map file.
478 if !linker.dynamicProperties.BuildStubs {
479 versionScript := ctx.ExpandOptionalSource(
Colin Crossc17727d2018-10-24 12:42:09 -0700480 linker.Properties.Version_script, "version_script")
Dan Albert61f32122018-07-26 14:00:24 -0700481
Colin Crossc17727d2018-10-24 12:42:09 -0700482 if ctx.useVndk() && linker.Properties.Target.Vendor.Version_script != nil {
Jiyong Park7ed9de32018-10-15 22:25:07 +0900483 versionScript = ctx.ExpandOptionalSource(
Colin Crossc17727d2018-10-24 12:42:09 -0700484 linker.Properties.Target.Vendor.Version_script,
Jiyong Park7ed9de32018-10-15 22:25:07 +0900485 "target.vendor.version_script")
486 }
Dan Albert61f32122018-07-26 14:00:24 -0700487
Jiyong Park7ed9de32018-10-15 22:25:07 +0900488 if versionScript.Valid() {
489 if ctx.Darwin() {
490 ctx.PropertyErrorf("version_script", "Not supported on Darwin")
491 } else {
Colin Cross4af21ed2019-11-04 09:37:55 -0800492 flags.Local.LdFlags = append(flags.Local.LdFlags,
Jiyong Park7ed9de32018-10-15 22:25:07 +0900493 "-Wl,--version-script,"+versionScript.String())
494 flags.LdFlagsDeps = append(flags.LdFlagsDeps, versionScript.Path())
495
496 if linker.sanitize.isSanitizerEnabled(cfi) {
497 cfiExportsMap := android.PathForSource(ctx, cfiExportsMapPath)
Colin Cross4af21ed2019-11-04 09:37:55 -0800498 flags.Local.LdFlags = append(flags.Local.LdFlags,
Jiyong Park7ed9de32018-10-15 22:25:07 +0900499 "-Wl,--version-script,"+cfiExportsMap.String())
500 flags.LdFlagsDeps = append(flags.LdFlagsDeps, cfiExportsMap)
501 }
Dan Albert61f32122018-07-26 14:00:24 -0700502 }
503 }
504 }
505
Colin Cross4d9c2d12016-07-29 12:48:20 -0700506 return flags
507}
508
Colin Crossb916a382016-07-29 17:28:03 -0700509func (linker *baseLinker) link(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700510 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Crossb916a382016-07-29 17:28:03 -0700511 panic(fmt.Errorf("baseLinker doesn't know how to link"))
Colin Cross4d9c2d12016-07-29 12:48:20 -0700512}
Colin Cross86803cf2018-02-15 14:12:26 -0800513
Paul Duffin13f02712020-03-06 12:30:43 +0000514func (linker *baseLinker) linkerSpecifiedDeps(specifiedDeps specifiedDeps) specifiedDeps {
515 specifiedDeps.sharedLibs = append(specifiedDeps.sharedLibs, linker.Properties.Shared_libs...)
Martin Stjernholm10566a02020-03-24 01:19:52 +0000516
517 // Must distinguish nil and [] in system_shared_libs - ensure that [] in
518 // either input list doesn't come out as nil.
519 if specifiedDeps.systemSharedLibs == nil {
520 specifiedDeps.systemSharedLibs = linker.Properties.System_shared_libs
521 } else {
522 specifiedDeps.systemSharedLibs = append(specifiedDeps.systemSharedLibs, linker.Properties.System_shared_libs...)
523 }
524
Paul Duffin13f02712020-03-06 12:30:43 +0000525 return specifiedDeps
526}
527
Colin Cross86803cf2018-02-15 14:12:26 -0800528// Injecting version symbols
529// Some host modules want a version number, but we don't want to rebuild it every time. Optionally add a step
530// after linking that injects a constant placeholder with the current version number.
531
532func init() {
533 pctx.HostBinToolVariable("symbolInjectCmd", "symbol_inject")
534}
535
536var injectVersionSymbol = pctx.AndroidStaticRule("injectVersionSymbol",
537 blueprint.RuleParams{
538 Command: "$symbolInjectCmd -i $in -o $out -s soong_build_number " +
Colin Cross2a2e0db2020-02-21 16:55:46 -0800539 "-from 'SOONG BUILD NUMBER PLACEHOLDER' -v $$(cat $buildNumberFile)",
Colin Cross86803cf2018-02-15 14:12:26 -0800540 CommandDeps: []string{"$symbolInjectCmd"},
541 },
Colin Cross2a2e0db2020-02-21 16:55:46 -0800542 "buildNumberFile")
Colin Cross86803cf2018-02-15 14:12:26 -0800543
544func (linker *baseLinker) injectVersionSymbol(ctx ModuleContext, in android.Path, out android.WritablePath) {
Colin Cross2a2e0db2020-02-21 16:55:46 -0800545 buildNumberFile := ctx.Config().BuildNumberFile(ctx)
Colin Cross86803cf2018-02-15 14:12:26 -0800546 ctx.Build(pctx, android.BuildParams{
547 Rule: injectVersionSymbol,
548 Description: "inject version symbol",
549 Input: in,
550 Output: out,
Colin Cross2a2e0db2020-02-21 16:55:46 -0800551 OrderOnly: android.Paths{buildNumberFile},
Colin Cross86803cf2018-02-15 14:12:26 -0800552 Args: map[string]string{
Colin Cross2a2e0db2020-02-21 16:55:46 -0800553 "buildNumberFile": buildNumberFile.String(),
Colin Cross86803cf2018-02-15 14:12:26 -0800554 },
555 })
556}
Vic Yang6cd1be82019-06-24 16:08:48 -0700557
558// Rule to generate .bss symbol ordering file.
559
560var (
561 _ = pctx.SourcePathVariable("genSortedBssSymbolsPath", "build/soong/scripts/gen_sorted_bss_symbols.sh")
562 gen_sorted_bss_symbols = pctx.AndroidStaticRule("gen_sorted_bss_symbols",
563 blueprint.RuleParams{
564 Command: "CROSS_COMPILE=$crossCompile $genSortedBssSymbolsPath ${in} ${out}",
Dan Willemsen724ab5d2019-09-19 10:50:18 -0700565 CommandDeps: []string{"$genSortedBssSymbolsPath", "${crossCompile}nm"},
Vic Yang6cd1be82019-06-24 16:08:48 -0700566 },
567 "crossCompile")
568)
569
570func (linker *baseLinker) sortBssSymbolsBySize(ctx ModuleContext, in android.Path, symbolOrderingFile android.ModuleOutPath, flags builderFlags) string {
571 crossCompile := gccCmd(flags.toolchain, "")
572 ctx.Build(pctx, android.BuildParams{
573 Rule: gen_sorted_bss_symbols,
574 Description: "generate bss symbol order " + symbolOrderingFile.Base(),
575 Output: symbolOrderingFile,
576 Input: in,
577 Args: map[string]string{
578 "crossCompile": crossCompile,
579 },
580 })
581 return "-Wl,--symbol-ordering-file," + symbolOrderingFile.String()
582}