Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 1 | // 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 | |
| 15 | package cc |
| 16 | |
| 17 | import ( |
| 18 | "github.com/google/blueprint" |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 19 | "github.com/google/blueprint/proptools" |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 20 | |
| 21 | "android/soong" |
| 22 | "android/soong/android" |
| 23 | ) |
| 24 | |
| 25 | type BinaryLinkerProperties struct { |
| 26 | // compile executable with -static |
| 27 | Static_executable *bool `android:"arch_variant"` |
| 28 | |
| 29 | // set the name of the output |
| 30 | Stem string `android:"arch_variant"` |
| 31 | |
| 32 | // append to the name of the output |
| 33 | Suffix string `android:"arch_variant"` |
| 34 | |
| 35 | // if set, add an extra objcopy --prefix-symbols= step |
| 36 | Prefix_symbols string |
| 37 | } |
| 38 | |
| 39 | func init() { |
| 40 | soong.RegisterModuleType("cc_binary", binaryFactory) |
| 41 | soong.RegisterModuleType("cc_binary_host", binaryHostFactory) |
| 42 | } |
| 43 | |
| 44 | // Module factory for binaries |
| 45 | func binaryFactory() (blueprint.Module, []interface{}) { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 46 | module, _ := NewBinary(android.HostAndDeviceSupported) |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 47 | return module.Init() |
| 48 | } |
| 49 | |
| 50 | // Module factory for host binaries |
| 51 | func binaryHostFactory() (blueprint.Module, []interface{}) { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 52 | module, _ := NewBinary(android.HostSupported) |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 53 | return module.Init() |
| 54 | } |
| 55 | |
| 56 | // |
| 57 | // Executables |
| 58 | // |
| 59 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 60 | type binaryDecorator struct { |
| 61 | *baseLinker |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 62 | stripper |
| 63 | |
| 64 | Properties BinaryLinkerProperties |
| 65 | |
| 66 | hostToolPath android.OptionalPath |
| 67 | } |
| 68 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 69 | var _ linker = (*binaryDecorator)(nil) |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 70 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 71 | func (binary *binaryDecorator) linkerProps() []interface{} { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 72 | return append(binary.baseLinker.linkerProps(), |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 73 | &binary.Properties, |
| 74 | &binary.stripper.StripProperties) |
| 75 | |
| 76 | } |
| 77 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 78 | func (binary *binaryDecorator) getStem(ctx BaseModuleContext) string { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 79 | stem := ctx.ModuleName() |
| 80 | if binary.Properties.Stem != "" { |
| 81 | stem = binary.Properties.Stem |
| 82 | } |
| 83 | |
| 84 | return stem + binary.Properties.Suffix |
| 85 | } |
| 86 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 87 | func (binary *binaryDecorator) linkerDeps(ctx BaseModuleContext, deps Deps) Deps { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 88 | deps = binary.baseLinker.linkerDeps(ctx, deps) |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 89 | if ctx.Device() { |
| 90 | if !Bool(binary.baseLinker.Properties.Nocrt) { |
| 91 | if !ctx.sdk() { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 92 | if binary.static() { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 93 | deps.CrtBegin = "crtbegin_static" |
| 94 | } else { |
| 95 | deps.CrtBegin = "crtbegin_dynamic" |
| 96 | } |
| 97 | deps.CrtEnd = "crtend_android" |
| 98 | } else { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 99 | if binary.static() { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 100 | deps.CrtBegin = "ndk_crtbegin_static." + ctx.sdkVersion() |
| 101 | } else { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 102 | if binary.static() { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 103 | deps.CrtBegin = "ndk_crtbegin_static." + ctx.sdkVersion() |
| 104 | } else { |
| 105 | deps.CrtBegin = "ndk_crtbegin_dynamic." + ctx.sdkVersion() |
| 106 | } |
| 107 | deps.CrtEnd = "ndk_crtend_android." + ctx.sdkVersion() |
| 108 | } |
| 109 | } |
| 110 | } |
| 111 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 112 | if binary.static() { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 113 | if inList("libc++_static", deps.StaticLibs) { |
| 114 | deps.StaticLibs = append(deps.StaticLibs, "libm", "libc", "libdl") |
| 115 | } |
| 116 | // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with |
| 117 | // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs, |
| 118 | // move them to the beginning of deps.LateStaticLibs |
| 119 | var groupLibs []string |
| 120 | deps.StaticLibs, groupLibs = filterList(deps.StaticLibs, |
| 121 | []string{"libc", "libc_nomalloc", "libcompiler_rt"}) |
| 122 | deps.LateStaticLibs = append(groupLibs, deps.LateStaticLibs...) |
| 123 | } |
| 124 | } |
| 125 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 126 | if !binary.static() && inList("libc", deps.StaticLibs) { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 127 | ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" + |
| 128 | "from static libs or set static_executable: true") |
| 129 | } |
| 130 | return deps |
| 131 | } |
| 132 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 133 | func (binary *binaryDecorator) isDependencyRoot() bool { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 134 | return true |
| 135 | } |
| 136 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 137 | func NewBinary(hod android.HostOrDeviceSupported) (*Module, *binaryDecorator) { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 138 | module := newModule(hod, android.MultilibFirst) |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 139 | binary := &binaryDecorator{ |
| 140 | baseLinker: NewBaseLinker(), |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 141 | } |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 142 | module.compiler = NewBaseCompiler() |
| 143 | module.linker = binary |
| 144 | module.installer = NewBaseInstaller("bin", "", InstallInSystem) |
| 145 | return module, binary |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 146 | } |
| 147 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 148 | func (binary *binaryDecorator) linkerInit(ctx BaseModuleContext) { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 149 | binary.baseLinker.linkerInit(ctx) |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 150 | |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 151 | if ctx.Host() { |
| 152 | if ctx.Os() == android.Linux { |
| 153 | if binary.Properties.Static_executable == nil && Bool(ctx.AConfig().ProductVariables.HostStaticBinaries) { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 154 | binary.Properties.Static_executable = proptools.BoolPtr(true) |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 155 | } |
| 156 | } else { |
| 157 | // Static executables are not supported on Darwin or Windows |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 158 | binary.Properties.Static_executable = nil |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 159 | } |
| 160 | } |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 161 | } |
| 162 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 163 | func (binary *binaryDecorator) static() bool { |
| 164 | return Bool(binary.Properties.Static_executable) |
| 165 | } |
| 166 | |
| 167 | func (binary *binaryDecorator) staticBinary() bool { |
| 168 | return binary.static() |
| 169 | } |
| 170 | |
| 171 | func (binary *binaryDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 172 | flags = binary.baseLinker.linkerFlags(ctx, flags) |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 173 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 174 | if ctx.Host() && !binary.static() { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 175 | flags.LdFlags = append(flags.LdFlags, "-pie") |
| 176 | if ctx.Os() == android.Windows { |
| 177 | flags.LdFlags = append(flags.LdFlags, "-Wl,-e_mainCRTStartup") |
| 178 | } |
| 179 | } |
| 180 | |
| 181 | // MinGW spits out warnings about -fPIC even for -fpie?!) being ignored because |
| 182 | // all code is position independent, and then those warnings get promoted to |
| 183 | // errors. |
| 184 | if ctx.Os() != android.Windows { |
| 185 | flags.CFlags = append(flags.CFlags, "-fpie") |
| 186 | } |
| 187 | |
| 188 | if ctx.Device() { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 189 | if binary.static() { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 190 | // Clang driver needs -static to create static executable. |
| 191 | // However, bionic/linker uses -shared to overwrite. |
| 192 | // Linker for x86 targets does not allow coexistance of -static and -shared, |
| 193 | // so we add -static only if -shared is not used. |
| 194 | if !inList("-shared", flags.LdFlags) { |
| 195 | flags.LdFlags = append(flags.LdFlags, "-static") |
| 196 | } |
| 197 | |
| 198 | flags.LdFlags = append(flags.LdFlags, |
| 199 | "-nostdlib", |
| 200 | "-Bstatic", |
| 201 | "-Wl,--gc-sections", |
| 202 | ) |
| 203 | |
| 204 | } else { |
| 205 | if flags.DynamicLinker == "" { |
| 206 | flags.DynamicLinker = "/system/bin/linker" |
| 207 | if flags.Toolchain.Is64Bit() { |
| 208 | flags.DynamicLinker += "64" |
| 209 | } |
| 210 | } |
| 211 | |
| 212 | flags.LdFlags = append(flags.LdFlags, |
| 213 | "-pie", |
| 214 | "-nostdlib", |
| 215 | "-Bdynamic", |
| 216 | "-Wl,--gc-sections", |
| 217 | "-Wl,-z,nocopyreloc", |
| 218 | ) |
| 219 | } |
| 220 | } else { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 221 | if binary.static() { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 222 | flags.LdFlags = append(flags.LdFlags, "-static") |
| 223 | } |
| 224 | if ctx.Darwin() { |
| 225 | flags.LdFlags = append(flags.LdFlags, "-Wl,-headerpad_max_install_names") |
| 226 | } |
| 227 | } |
| 228 | |
| 229 | return flags |
| 230 | } |
| 231 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 232 | func (binary *binaryDecorator) link(ctx ModuleContext, |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 233 | flags Flags, deps PathDeps, objFiles android.Paths) android.Path { |
| 234 | |
| 235 | fileName := binary.getStem(ctx) + flags.Toolchain.ExecutableSuffix() |
| 236 | outputFile := android.PathForModuleOut(ctx, fileName) |
| 237 | ret := outputFile |
| 238 | if ctx.Os().Class == android.Host { |
| 239 | binary.hostToolPath = android.OptionalPathForPath(outputFile) |
| 240 | } |
| 241 | |
| 242 | var linkerDeps android.Paths |
| 243 | |
| 244 | sharedLibs := deps.SharedLibs |
| 245 | sharedLibs = append(sharedLibs, deps.LateSharedLibs...) |
| 246 | |
| 247 | if flags.DynamicLinker != "" { |
| 248 | flags.LdFlags = append(flags.LdFlags, " -Wl,-dynamic-linker,"+flags.DynamicLinker) |
| 249 | } |
| 250 | |
| 251 | builderFlags := flagsToBuilderFlags(flags) |
| 252 | |
| 253 | if binary.stripper.needsStrip(ctx) { |
| 254 | strippedOutputFile := outputFile |
| 255 | outputFile = android.PathForModuleOut(ctx, "unstripped", fileName) |
| 256 | binary.stripper.strip(ctx, outputFile, strippedOutputFile, builderFlags) |
| 257 | } |
| 258 | |
| 259 | if binary.Properties.Prefix_symbols != "" { |
| 260 | afterPrefixSymbols := outputFile |
| 261 | outputFile = android.PathForModuleOut(ctx, "unprefixed", fileName) |
| 262 | TransformBinaryPrefixSymbols(ctx, binary.Properties.Prefix_symbols, outputFile, |
| 263 | flagsToBuilderFlags(flags), afterPrefixSymbols) |
| 264 | } |
| 265 | |
| 266 | TransformObjToDynamicBinary(ctx, objFiles, sharedLibs, deps.StaticLibs, |
| 267 | deps.LateStaticLibs, deps.WholeStaticLibs, linkerDeps, deps.CrtBegin, deps.CrtEnd, true, |
| 268 | builderFlags, outputFile) |
| 269 | |
| 270 | return ret |
| 271 | } |
| 272 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame^] | 273 | func (binary *binaryDecorator) HostToolPath() android.OptionalPath { |
Colin Cross | 4d9c2d1 | 2016-07-29 12:48:20 -0700 | [diff] [blame] | 274 | return binary.hostToolPath |
| 275 | } |