blob: bcedc8da14b437b85660352bfb2549fe4863b6f3 [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"
19 "fmt"
Colin Cross4b963f82016-09-29 14:06:02 -070020
Colin Cross86803cf2018-02-15 14:12:26 -080021 "github.com/google/blueprint"
Colin Cross4b963f82016-09-29 14:06:02 -070022 "github.com/google/blueprint/proptools"
Colin Crossb916a382016-07-29 17:28:03 -070023)
24
Colin Cross4d9c2d12016-07-29 12:48:20 -070025// This file contains the basic functionality for linking against static libraries and shared
26// libraries. Final linking into libraries or executables is handled in library.go, binary.go, etc.
27
28type BaseLinkerProperties struct {
29 // list of modules whose object files should be linked into this module
30 // in their entirety. For static library modules, all of the .o files from the intermediate
31 // directory of the dependency will be linked into this modules .a file. For a shared library,
32 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
33 Whole_static_libs []string `android:"arch_variant,variant_prepend"`
34
35 // list of modules that should be statically linked into this module.
36 Static_libs []string `android:"arch_variant,variant_prepend"`
37
38 // list of modules that should be dynamically linked into this module.
39 Shared_libs []string `android:"arch_variant"`
40
Colin Cross5950f382016-12-13 12:50:57 -080041 // list of modules that should only provide headers for this module.
42 Header_libs []string `android:"arch_variant,variant_prepend"`
43
Colin Cross4d9c2d12016-07-29 12:48:20 -070044 // list of module-specific flags that will be used for all link steps
45 Ldflags []string `android:"arch_variant"`
46
Colin Cross4d9c2d12016-07-29 12:48:20 -070047 // list of system libraries that will be dynamically linked to
Colin Crossef88ae22017-08-18 16:52:25 -070048 // shared library and executable modules. If unset, generally defaults to libc,
49 // libm, and libdl. Set to [] to prevent linking against the defaults.
Colin Cross4d9c2d12016-07-29 12:48:20 -070050 System_shared_libs []string
51
52 // allow the module to contain undefined symbols. By default,
53 // modules cannot contain undefined symbols that are not satisified by their immediate
54 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
55 // This flag should only be necessary for compiling low-level libraries like libc.
Colin Crossbe360ae2016-12-08 09:45:21 -080056 Allow_undefined_symbols *bool `android:"arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -070057
58 // don't link in libgcc.a
59 No_libgcc *bool
60
61 // -l arguments to pass to linker for host-provided shared libraries
62 Host_ldlibs []string `android:"arch_variant"`
63
64 // list of shared libraries to re-export include directories from. Entries must be
65 // present in shared_libs.
66 Export_shared_lib_headers []string `android:"arch_variant"`
67
68 // list of static libraries to re-export include directories from. Entries must be
69 // present in static_libs.
70 Export_static_lib_headers []string `android:"arch_variant"`
71
Colin Cross5950f382016-12-13 12:50:57 -080072 // list of header libraries to re-export include directories from. Entries must be
73 // present in header_libs.
74 Export_header_lib_headers []string `android:"arch_variant"`
75
Dan Willemsenb3454ab2016-09-28 17:34:58 -070076 // list of generated headers to re-export include directories from. Entries must be
77 // present in generated_headers.
78 Export_generated_headers []string `android:"arch_variant"`
79
Colin Cross4d9c2d12016-07-29 12:48:20 -070080 // don't link in crt_begin and crt_end. This flag should only be necessary for
81 // compiling crt or libc.
82 Nocrt *bool `android:"arch_variant"`
Colin Cross18c0c5a2016-12-01 14:45:23 -080083
84 // group static libraries. This can resolve missing symbols issues with interdependencies
85 // between static libraries, but it is generally better to order them correctly instead.
86 Group_static_libs *bool `android:"arch_variant"`
Jiyong Park44cf1a72017-06-16 12:03:55 +090087
88 Target struct {
89 Vendor struct {
90 // list of shared libs that should not be used to build
91 // the vendor variant of the C/C++ module.
92 Exclude_shared_libs []string
Jiyong Park52d25bd2017-10-13 09:17:01 +090093
94 // list of static libs that should not be used to build
95 // the vendor variant of the C/C++ module.
96 Exclude_static_libs []string
Jiyong Park44cf1a72017-06-16 12:03:55 +090097 }
98 }
Colin Cross86803cf2018-02-15 14:12:26 -080099
100 // make android::build:GetBuildNumber() available containing the build ID.
101 Use_version_lib *bool `android:"arch_variant"`
Colin Cross4d9c2d12016-07-29 12:48:20 -0700102}
103
Colin Crossb916a382016-07-29 17:28:03 -0700104func NewBaseLinker() *baseLinker {
105 return &baseLinker{}
106}
107
Colin Cross4d9c2d12016-07-29 12:48:20 -0700108// baseLinker provides support for shared_libs, static_libs, and whole_static_libs properties
109type baseLinker struct {
110 Properties BaseLinkerProperties
111 dynamicProperties struct {
Colin Crossb916a382016-07-29 17:28:03 -0700112 RunPaths []string `blueprint:"mutated"`
Colin Cross4d9c2d12016-07-29 12:48:20 -0700113 }
114}
115
116func (linker *baseLinker) appendLdflags(flags []string) {
117 linker.Properties.Ldflags = append(linker.Properties.Ldflags, flags...)
118}
119
Colin Cross42742b82016-08-01 13:20:05 -0700120func (linker *baseLinker) linkerInit(ctx BaseModuleContext) {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700121 if ctx.toolchain().Is64Bit() {
Colin Crossb916a382016-07-29 17:28:03 -0700122 linker.dynamicProperties.RunPaths = append(linker.dynamicProperties.RunPaths, "../lib64", "lib64")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700123 } else {
Colin Crossb916a382016-07-29 17:28:03 -0700124 linker.dynamicProperties.RunPaths = append(linker.dynamicProperties.RunPaths, "../lib", "lib")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700125 }
126}
127
Colin Cross42742b82016-08-01 13:20:05 -0700128func (linker *baseLinker) linkerProps() []interface{} {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700129 return []interface{}{&linker.Properties, &linker.dynamicProperties}
130}
131
Colin Cross42742b82016-08-01 13:20:05 -0700132func (linker *baseLinker) linkerDeps(ctx BaseModuleContext, deps Deps) Deps {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700133 deps.WholeStaticLibs = append(deps.WholeStaticLibs, linker.Properties.Whole_static_libs...)
Colin Cross5950f382016-12-13 12:50:57 -0800134 deps.HeaderLibs = append(deps.HeaderLibs, linker.Properties.Header_libs...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700135 deps.StaticLibs = append(deps.StaticLibs, linker.Properties.Static_libs...)
136 deps.SharedLibs = append(deps.SharedLibs, linker.Properties.Shared_libs...)
137
Colin Cross5950f382016-12-13 12:50:57 -0800138 deps.ReexportHeaderLibHeaders = append(deps.ReexportHeaderLibHeaders, linker.Properties.Export_header_lib_headers...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700139 deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, linker.Properties.Export_static_lib_headers...)
140 deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, linker.Properties.Export_shared_lib_headers...)
Dan Willemsenb3454ab2016-09-28 17:34:58 -0700141 deps.ReexportGeneratedHeaders = append(deps.ReexportGeneratedHeaders, linker.Properties.Export_generated_headers...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700142
Colin Cross86803cf2018-02-15 14:12:26 -0800143 if Bool(linker.Properties.Use_version_lib) {
144 deps.WholeStaticLibs = append(deps.WholeStaticLibs, "libbuildversion")
145 }
146
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700147 if ctx.useVndk() {
Jiyong Park74472282017-08-10 00:48:06 +0900148 deps.SharedLibs = removeListFromList(deps.SharedLibs, linker.Properties.Target.Vendor.Exclude_shared_libs)
149 deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, linker.Properties.Target.Vendor.Exclude_shared_libs)
Jiyong Park52d25bd2017-10-13 09:17:01 +0900150 deps.StaticLibs = removeListFromList(deps.StaticLibs, linker.Properties.Target.Vendor.Exclude_static_libs)
151 deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, linker.Properties.Target.Vendor.Exclude_static_libs)
152 deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, linker.Properties.Target.Vendor.Exclude_static_libs)
Jiyong Park74472282017-08-10 00:48:06 +0900153 }
154
Dan Albert83705c82016-09-14 16:47:18 -0700155 if ctx.ModuleName() != "libcompiler_rt-extras" {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700156 deps.LateStaticLibs = append(deps.LateStaticLibs, "libcompiler_rt-extras")
157 }
158
Dan Willemsen2e47b342016-11-17 01:02:25 -0800159 if ctx.toolchain().Bionic() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700160 // libgcc and libatomic have to be last on the command line
161 deps.LateStaticLibs = append(deps.LateStaticLibs, "libatomic")
162 if !Bool(linker.Properties.No_libgcc) {
163 deps.LateStaticLibs = append(deps.LateStaticLibs, "libgcc")
164 }
165
Colin Crossb916a382016-07-29 17:28:03 -0700166 if !ctx.static() {
Colin Crossef88ae22017-08-18 16:52:25 -0700167 systemSharedLibs := linker.Properties.System_shared_libs
168 if systemSharedLibs == nil {
169 systemSharedLibs = []string{"libc", "libm", "libdl"}
Dimitry Ivanovba6b5522017-06-26 18:13:56 -0700170 }
171
Colin Crossef88ae22017-08-18 16:52:25 -0700172 if inList("libdl", deps.SharedLibs) {
173 // If system_shared_libs has libc but not libdl, make sure shared_libs does not
174 // have libdl to avoid loading libdl before libc.
175 if inList("libc", systemSharedLibs) {
176 if !inList("libdl", systemSharedLibs) {
177 ctx.PropertyErrorf("shared_libs",
178 "libdl must be in system_shared_libs, not shared_libs")
179 }
180 _, deps.SharedLibs = removeFromList("libdl", deps.SharedLibs)
Dimitry Ivanovba6b5522017-06-26 18:13:56 -0700181 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700182 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700183
Colin Crossef88ae22017-08-18 16:52:25 -0700184 // If libc and libdl are both in system_shared_libs make sure libd comes after libc
185 // to avoid loading libdl before libc.
186 if inList("libdl", systemSharedLibs) && inList("libc", systemSharedLibs) &&
187 indexList("libdl", systemSharedLibs) < indexList("libc", systemSharedLibs) {
188 ctx.PropertyErrorf("system_shared_libs", "libdl must be after libc")
189 }
190
191 deps.LateSharedLibs = append(deps.LateSharedLibs, systemSharedLibs...)
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700192 } else if ctx.useSdk() || ctx.useVndk() {
Dimitry Ivanovba6b5522017-06-26 18:13:56 -0700193 deps.LateSharedLibs = append(deps.LateSharedLibs, "libc", "libm", "libdl")
Dan Willemsen4416e5d2017-04-06 12:43:22 -0700194 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700195 }
196
Colin Cross3edeee12017-04-04 12:59:48 -0700197 if ctx.Windows() {
Josh Gao7bd4c5c2017-02-23 17:52:24 -0800198 deps.LateStaticLibs = append(deps.LateStaticLibs, "libwinpthread")
199 }
200
Colin Cross4d9c2d12016-07-29 12:48:20 -0700201 return deps
202}
203
Colin Cross42742b82016-08-01 13:20:05 -0700204func (linker *baseLinker) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700205 toolchain := ctx.toolchain()
206
Colin Cross324a4572017-11-02 23:09:41 -0700207 hod := "Host"
208 if ctx.Os().Class == android.Device {
209 hod = "Device"
210 }
211
Colin Cross87dd9632017-11-03 13:31:05 -0700212 flags.LdFlags = append(flags.LdFlags, fmt.Sprintf("${config.%sGlobalLdflags}", hod))
213 if Bool(linker.Properties.Allow_undefined_symbols) {
214 if ctx.Darwin() {
215 // darwin defaults to treating undefined symbols as errors
216 flags.LdFlags = append(flags.LdFlags, "-Wl,-undefined,dynamic_lookup")
Colin Cross4d9c2d12016-07-29 12:48:20 -0700217 }
Colin Cross87dd9632017-11-03 13:31:05 -0700218 } else if !ctx.Darwin() {
219 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
220 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700221
Colin Cross87dd9632017-11-03 13:31:05 -0700222 if flags.Clang {
223 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
224 } else {
225 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
226 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700227
Colin Cross87dd9632017-11-03 13:31:05 -0700228 if !ctx.toolchain().Bionic() {
229 CheckBadHostLdlibs(ctx, "host_ldlibs", linker.Properties.Host_ldlibs)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700230
Colin Cross87dd9632017-11-03 13:31:05 -0700231 flags.LdFlags = append(flags.LdFlags, linker.Properties.Host_ldlibs...)
Colin Crossc5fdbb82017-09-08 12:45:18 -0700232
Colin Cross87dd9632017-11-03 13:31:05 -0700233 if !ctx.Windows() {
234 // Add -ldl, -lpthread, -lm and -lrt to host builds to match the default behavior of device
235 // builds
236 flags.LdFlags = append(flags.LdFlags,
237 "-ldl",
238 "-lpthread",
239 "-lm",
240 )
241 if !ctx.Darwin() {
242 flags.LdFlags = append(flags.LdFlags, "-lrt")
Colin Crossc5fdbb82017-09-08 12:45:18 -0700243 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700244 }
245 }
246
247 CheckBadLinkerFlags(ctx, "ldflags", linker.Properties.Ldflags)
248
Colin Cross4b963f82016-09-29 14:06:02 -0700249 flags.LdFlags = append(flags.LdFlags, proptools.NinjaAndShellEscape(linker.Properties.Ldflags)...)
Colin Cross4d9c2d12016-07-29 12:48:20 -0700250
Colin Crossb916a382016-07-29 17:28:03 -0700251 if ctx.Host() {
Colin Cross4d9c2d12016-07-29 12:48:20 -0700252 rpath_prefix := `\$$ORIGIN/`
253 if ctx.Darwin() {
254 rpath_prefix = "@loader_path/"
255 }
256
Dan Willemsen6f91fbf2017-09-18 22:45:15 -0700257 if !ctx.static() {
258 for _, rpath := range linker.dynamicProperties.RunPaths {
259 flags.LdFlags = append(flags.LdFlags, "-Wl,-rpath,"+rpath_prefix+rpath)
260 }
Colin Cross4d9c2d12016-07-29 12:48:20 -0700261 }
262 }
263
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700264 if ctx.useSdk() && (ctx.Arch().ArchType != android.Mips && ctx.Arch().ArchType != android.Mips64) {
Colin Cross6774e282017-08-11 13:23:57 -0700265 // The bionic linker now has support gnu style hashes (which are much faster!), but shipping
266 // to older devices requires the old style hash. Fortunately, we can build with both and
267 // it'll work anywhere.
268 // This is not currently supported on MIPS architectures.
269 flags.LdFlags = append(flags.LdFlags, "-Wl,--hash-style=both")
270 }
271
Colin Cross4d9c2d12016-07-29 12:48:20 -0700272 if flags.Clang {
273 flags.LdFlags = append(flags.LdFlags, toolchain.ToolchainClangLdflags())
274 } else {
275 flags.LdFlags = append(flags.LdFlags, toolchain.ToolchainLdflags())
276 }
277
Colin Cross18c0c5a2016-12-01 14:45:23 -0800278 if Bool(linker.Properties.Group_static_libs) {
279 flags.GroupStaticLibs = true
280 }
281
Colin Cross4d9c2d12016-07-29 12:48:20 -0700282 return flags
283}
284
Colin Crossb916a382016-07-29 17:28:03 -0700285func (linker *baseLinker) link(ctx ModuleContext,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700286 flags Flags, deps PathDeps, objs Objects) android.Path {
Colin Crossb916a382016-07-29 17:28:03 -0700287 panic(fmt.Errorf("baseLinker doesn't know how to link"))
Colin Cross4d9c2d12016-07-29 12:48:20 -0700288}
Colin Cross86803cf2018-02-15 14:12:26 -0800289
290// Injecting version symbols
291// Some host modules want a version number, but we don't want to rebuild it every time. Optionally add a step
292// after linking that injects a constant placeholder with the current version number.
293
294func init() {
295 pctx.HostBinToolVariable("symbolInjectCmd", "symbol_inject")
296}
297
298var injectVersionSymbol = pctx.AndroidStaticRule("injectVersionSymbol",
299 blueprint.RuleParams{
300 Command: "$symbolInjectCmd -i $in -o $out -s soong_build_number " +
301 "-from 'SOONG BUILD NUMBER PLACEHOLDER' -v $buildNumberFromFile",
302 CommandDeps: []string{"$symbolInjectCmd"},
303 },
304 "buildNumberFromFile")
305
306func (linker *baseLinker) injectVersionSymbol(ctx ModuleContext, in android.Path, out android.WritablePath) {
307 ctx.Build(pctx, android.BuildParams{
308 Rule: injectVersionSymbol,
309 Description: "inject version symbol",
310 Input: in,
311 Output: out,
312 Args: map[string]string{
313 "buildNumberFromFile": ctx.Config().BuildNumberFromFile(),
314 },
315 })
316}