blob: 3ab8a4764113911c4fadf5d0cc91274a23c25df0 [file] [log] [blame]
Colin Cross5049f022015-03-18 13:28:46 -07001// Copyright 2015 Google Inc. All rights reserved.
Colin Cross3f40fa42015-01-30 17:27:36 -08002//
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
17// This file contains the module types for compiling C/C++ for Android, and converts the properties
18// into the flags and filenames necessary to pass to the compiler. The final creation of the rules
19// is handled in builder.go
20
21import (
Colin Cross3f40fa42015-01-30 17:27:36 -080022 "fmt"
23 "path/filepath"
24 "strings"
25
Colin Cross97ba0732015-03-23 17:50:24 -070026 "github.com/google/blueprint"
27 "github.com/google/blueprint/pathtools"
28
Colin Cross3f40fa42015-01-30 17:27:36 -080029 "android/soong/common"
Colin Cross5049f022015-03-18 13:28:46 -070030 "android/soong/genrule"
Colin Cross3f40fa42015-01-30 17:27:36 -080031)
32
Colin Cross3f40fa42015-01-30 17:27:36 -080033var (
Colin Cross1332b002015-04-07 17:11:30 -070034 HostPrebuiltTag = pctx.VariableConfigMethod("HostPrebuiltTag", common.Config.PrebuiltOS)
35 SrcDir = pctx.VariableConfigMethod("SrcDir", common.Config.SrcDir)
Colin Cross3f40fa42015-01-30 17:27:36 -080036
37 LibcRoot = pctx.StaticVariable("LibcRoot", "${SrcDir}/bionic/libc")
38 LibmRoot = pctx.StaticVariable("LibmRoot", "${SrcDir}/bionic/libm")
39)
40
41// Flags used by lots of devices. Putting them in package static variables will save bytes in
42// build.ninja so they aren't repeated for every file
43var (
44 commonGlobalCflags = []string{
45 "-DANDROID",
46 "-fmessage-length=0",
47 "-W",
48 "-Wall",
49 "-Wno-unused",
50 "-Winit-self",
51 "-Wpointer-arith",
52
53 // COMMON_RELEASE_CFLAGS
54 "-DNDEBUG",
55 "-UDEBUG",
56 }
57
58 deviceGlobalCflags = []string{
59 // TARGET_ERROR_FLAGS
60 "-Werror=return-type",
61 "-Werror=non-virtual-dtor",
62 "-Werror=address",
63 "-Werror=sequence-point",
64 }
65
66 hostGlobalCflags = []string{}
67
68 commonGlobalCppflags = []string{
69 "-Wsign-promo",
70 "-std=gnu++11",
71 }
72)
73
74func init() {
75 pctx.StaticVariable("commonGlobalCflags", strings.Join(commonGlobalCflags, " "))
76 pctx.StaticVariable("deviceGlobalCflags", strings.Join(deviceGlobalCflags, " "))
77 pctx.StaticVariable("hostGlobalCflags", strings.Join(hostGlobalCflags, " "))
78
79 pctx.StaticVariable("commonGlobalCppflags", strings.Join(commonGlobalCppflags, " "))
80
81 pctx.StaticVariable("commonClangGlobalCflags",
82 strings.Join(clangFilterUnknownCflags(commonGlobalCflags), " "))
83 pctx.StaticVariable("deviceClangGlobalCflags",
84 strings.Join(clangFilterUnknownCflags(deviceGlobalCflags), " "))
85 pctx.StaticVariable("hostClangGlobalCflags",
86 strings.Join(clangFilterUnknownCflags(hostGlobalCflags), " "))
Tim Kilbournf2948142015-03-11 12:03:03 -070087 pctx.StaticVariable("commonClangGlobalCppflags",
88 strings.Join(clangFilterUnknownCflags(commonGlobalCppflags), " "))
Colin Cross3f40fa42015-01-30 17:27:36 -080089
90 // Everything in this list is a crime against abstraction and dependency tracking.
91 // Do not add anything to this list.
92 pctx.StaticVariable("commonGlobalIncludes", strings.Join([]string{
93 "-isystem ${SrcDir}/system/core/include",
94 "-isystem ${SrcDir}/hardware/libhardware/include",
95 "-isystem ${SrcDir}/hardware/libhardware_legacy/include",
96 "-isystem ${SrcDir}/hardware/ril/include",
97 "-isystem ${SrcDir}/libnativehelper/include",
98 "-isystem ${SrcDir}/frameworks/native/include",
99 "-isystem ${SrcDir}/frameworks/native/opengl/include",
100 "-isystem ${SrcDir}/frameworks/av/include",
101 "-isystem ${SrcDir}/frameworks/base/include",
102 }, " "))
103
104 pctx.StaticVariable("clangPath", "${SrcDir}/prebuilts/clang/${HostPrebuiltTag}/host/3.6/bin/")
105}
106
Colin Cross3f40fa42015-01-30 17:27:36 -0800107// Building C/C++ code is handled by objects that satisfy this interface via composition
Colin Cross97ba0732015-03-23 17:50:24 -0700108type CCModuleType interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800109 common.AndroidModule
110
Colin Crossfa138792015-04-24 17:31:52 -0700111 // Modify property values after parsing Blueprints file but before starting dependency
112 // resolution or build rule generation
113 ModifyProperties(common.AndroidBaseContext)
114
Colin Cross21b9a242015-03-24 14:15:58 -0700115 // Modify the ccFlags
Colin Cross0676e2d2015-04-24 17:39:18 -0700116 flags(common.AndroidModuleContext, CCFlags) CCFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800117
Colin Cross21b9a242015-03-24 14:15:58 -0700118 // Return list of dependency names for use in AndroidDynamicDependencies and in depsToPaths
Colin Cross0676e2d2015-04-24 17:39:18 -0700119 depNames(common.AndroidBaseContext, CCDeps) CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800120
121 // Compile objects into final module
Colin Cross97ba0732015-03-23 17:50:24 -0700122 compileModule(common.AndroidModuleContext, CCFlags, CCDeps, []string)
Colin Cross3f40fa42015-01-30 17:27:36 -0800123
Dan Albertc403f7c2015-03-18 14:01:18 -0700124 // Install the built module.
Colin Cross97ba0732015-03-23 17:50:24 -0700125 installModule(common.AndroidModuleContext, CCFlags)
Dan Albertc403f7c2015-03-18 14:01:18 -0700126
Colin Cross3f40fa42015-01-30 17:27:36 -0800127 // Return the output file (.o, .a or .so) for use by other modules
128 outputFile() string
129}
130
Colin Cross97ba0732015-03-23 17:50:24 -0700131type CCDeps struct {
Colin Cross28344522015-04-22 13:07:53 -0700132 StaticLibs, SharedLibs, LateStaticLibs, WholeStaticLibs, ObjFiles, Cflags []string
Colin Crossc472d572015-03-17 15:06:21 -0700133
Colin Cross21b9a242015-03-24 14:15:58 -0700134 WholeStaticLibObjFiles []string
135
Colin Cross97ba0732015-03-23 17:50:24 -0700136 CrtBegin, CrtEnd string
Colin Crossc472d572015-03-17 15:06:21 -0700137}
138
Colin Cross97ba0732015-03-23 17:50:24 -0700139type CCFlags struct {
Colin Cross28344522015-04-22 13:07:53 -0700140 GlobalFlags []string // Flags that apply to C, C++, and assembly source files
141 AsFlags []string // Flags that apply to assembly source files
142 CFlags []string // Flags that apply to C and C++ source files
143 ConlyFlags []string // Flags that apply to C source files
144 CppFlags []string // Flags that apply to C++ source files
145 YaccFlags []string // Flags that apply to Yacc source files
146 LdFlags []string // Flags that apply to linker command lines
147
148 Nocrt bool
149 Toolchain Toolchain
150 Clang bool
Colin Crossc472d572015-03-17 15:06:21 -0700151}
152
Colin Crossfa138792015-04-24 17:31:52 -0700153// CCBase contains the properties and members used by all C/C++ module types, and implements
Colin Crossc472d572015-03-17 15:06:21 -0700154// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
155// and uses a ccModuleType interface to that struct to create the build steps.
Colin Crossfa138792015-04-24 17:31:52 -0700156type CCBase struct {
Colin Crossc472d572015-03-17 15:06:21 -0700157 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700158 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700159
Colin Crossfa138792015-04-24 17:31:52 -0700160 // Properties used to compile all C or C++ modules
161 Properties struct {
162 // srcs: list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
163 Srcs []string `android:"arch_variant,arch_subtract"`
164
165 // cflags: list of module-specific flags that will be used for C and C++ compiles.
166 Cflags []string `android:"arch_variant"`
167
168 // cppflags: list of module-specific flags that will be used for C++ compiles
169 Cppflags []string `android:"arch_variant"`
170
171 // conlyflags: list of module-specific flags that will be used for C compiles
172 Conlyflags []string `android:"arch_variant"`
173
174 // asflags: list of module-specific flags that will be used for .S compiles
175 Asflags []string `android:"arch_variant"`
176
177 // yaccflags: list of module-specific flags that will be used for .y and .yy compiles
178 Yaccflags []string
179
180 // ldflags: list of module-specific flags that will be used for all link steps
181 Ldflags []string `android:"arch_variant"`
182
183 // instruction_set: the instruction set architecture to use to compile the C/C++
184 // module.
185 Instruction_set string `android:"arch_variant"`
186
187 // include_dirs: list of directories relative to the root of the source tree that will
188 // be added to the include path using -I.
189 // If possible, don't use this. If adding paths from the current directory use
190 // local_include_dirs, if adding paths from other modules use export_include_dirs in
191 // that module.
192 Include_dirs []string `android:"arch_variant"`
193
194 // local_include_dirs: list of directories relative to the Blueprints file that will
195 // be added to the include path using -I
196 Local_include_dirs []string `android:"arch_variant"`
197
198 // export_include_dirs: list of directories relative to the Blueprints file that will
199 // be added to the include path using -I for any module that links against this module
200 Export_include_dirs []string `android:"arch_variant"`
201
202 // clang_cflags: list of module-specific flags that will be used for C and C++ compiles when
203 // compiling with clang
204 Clang_cflags []string `android:"arch_variant"`
205
206 // clang_asflags: list of module-specific flags that will be used for .S compiles when
207 // compiling with clang
208 Clang_asflags []string `android:"arch_variant"`
209
210 // system_shared_libs: list of system libraries that will be dynamically linked to
211 // shared library and executable modules. If unset, generally defaults to libc
212 // and libm. Set to [] to prevent linking against libc and libm.
213 System_shared_libs []string
214
215 // whole_static_libs: list of modules whose object files should be linked into this module
216 // in their entirety. For static library modules, all of the .o files from the intermediate
217 // directory of the dependency will be linked into this modules .a file. For a shared library,
218 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
219 Whole_static_libs []string `android:"arch_variant"`
220
221 // static_libs: list of modules that should be statically linked into this module.
222 Static_libs []string `android:"arch_variant"`
223
224 // shared_libs: list of modules that should be dynamically linked into this module.
225 Shared_libs []string `android:"arch_variant"`
226
227 // allow_undefined_symbols: allow the module to contain undefined symbols. By default,
228 // modules cannot contain undefined symbols that are not satisified by their immediate
229 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
230 // This flag should only be necessary for compiling low-level libraries like libc.
231 Allow_undefined_symbols bool
232
233 // nocrt: don't link in crt_begin and crt_end. This flag should only be necessary for
234 // compiling crt or libc.
235 Nocrt bool `android:"arch_variant"`
236
237 // no_default_compiler_flags: don't insert default compiler flags into asflags, cflags,
238 // cppflags, conlyflags, ldflags, or include_dirs
239 No_default_compiler_flags bool
240
241 // clang: compile module with clang instead of gcc
242 Clang bool `android:"arch_variant"`
243
244 // rtti: pass -frtti instead of -fno-rtti
245 Rtti bool
246
247 // host_ldlibs: -l arguments to pass to linker for host-provided shared libraries
248 Host_ldlibs []string `android:"arch_variant"`
249
250 // stl: select the STL library to use. Possible values are "libc++", "libc++_static",
251 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
252 // default
253 Stl string
254
255 // Set for combined shared/static libraries to prevent compiling object files a second time
256 SkipCompileObjs bool `blueprint:"mutated"`
257
258 Debug struct {
259 Cflags []string `android:"arch_variant"`
260 } `android:"arch_variant"`
261 Release struct {
262 Cflags []string `android:"arch_variant"`
263 } `android:"arch_variant"`
264
265 // Minimum sdk version supported when compiling against the ndk
266 Sdk_version string
267
268 // relative_install_path: install to a subdirectory of the default install path for the module
269 Relative_install_path string
270 }
271
272 unused struct {
273 Asan bool
274 Native_coverage bool
275 Strip string
276 Tags []string
277 Required []string
278 }
Colin Crossc472d572015-03-17 15:06:21 -0700279
280 installPath string
Colin Cross74d1ec02015-04-28 13:30:13 -0700281
282 savedDepNames CCDeps
Colin Crossc472d572015-03-17 15:06:21 -0700283}
284
Colin Crossfa138792015-04-24 17:31:52 -0700285func newCCBase(base *CCBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700286 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
287
288 base.module = module
289
Colin Crossfa138792015-04-24 17:31:52 -0700290 props = append(props, &base.Properties, &base.unused)
Colin Crossc472d572015-03-17 15:06:21 -0700291
Colin Cross5049f022015-03-18 13:28:46 -0700292 return common.InitAndroidArchModule(module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700293}
294
Colin Crossfa138792015-04-24 17:31:52 -0700295func (c *CCBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800296 toolchain := c.findToolchain(ctx)
297 if ctx.Failed() {
298 return
299 }
300
Colin Cross21b9a242015-03-24 14:15:58 -0700301 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800302 if ctx.Failed() {
303 return
304 }
305
Colin Cross74d1ec02015-04-28 13:30:13 -0700306 deps := c.depsToPaths(ctx, c.savedDepNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800307 if ctx.Failed() {
308 return
309 }
310
Colin Cross28344522015-04-22 13:07:53 -0700311 flags.CFlags = append(flags.CFlags, deps.Cflags...)
Colin Crossed9f8682015-03-18 17:17:35 -0700312
Colin Cross581c1892015-04-07 16:50:10 -0700313 objFiles := c.compileObjs(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800314 if ctx.Failed() {
315 return
316 }
317
Colin Cross581c1892015-04-07 16:50:10 -0700318 generatedObjFiles := c.compileGeneratedObjs(ctx, flags)
Colin Cross5049f022015-03-18 13:28:46 -0700319 if ctx.Failed() {
320 return
321 }
322
323 objFiles = append(objFiles, generatedObjFiles...)
324
Colin Cross3f40fa42015-01-30 17:27:36 -0800325 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
326 if ctx.Failed() {
327 return
328 }
Dan Albertc403f7c2015-03-18 14:01:18 -0700329
330 c.ccModuleType().installModule(ctx, flags)
331 if ctx.Failed() {
332 return
333 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800334}
335
Colin Crossfa138792015-04-24 17:31:52 -0700336func (c *CCBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800337 return c.module
338}
339
Colin Crossfa138792015-04-24 17:31:52 -0700340var _ common.AndroidDynamicDepender = (*CCBase)(nil)
Colin Cross3f40fa42015-01-30 17:27:36 -0800341
Colin Crossfa138792015-04-24 17:31:52 -0700342func (c *CCBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800343 arch := ctx.Arch()
344 factory := toolchainFactories[arch.HostOrDevice][arch.ArchType]
345 if factory == nil {
346 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
347 arch.HostOrDevice.String(), arch.String()))
348 }
349 return factory(arch.ArchVariant, arch.CpuVariant)
350}
351
Colin Crossfa138792015-04-24 17:31:52 -0700352func (c *CCBase) ModifyProperties(ctx common.AndroidBaseContext) {
353}
354
Colin Crosse11befc2015-04-27 17:49:17 -0700355func (c *CCBase) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossfa138792015-04-24 17:31:52 -0700356 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.Properties.Whole_static_libs...)
357 depNames.StaticLibs = append(depNames.StaticLibs, c.Properties.Static_libs...)
358 depNames.SharedLibs = append(depNames.SharedLibs, c.Properties.Shared_libs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700359
Colin Cross21b9a242015-03-24 14:15:58 -0700360 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800361}
362
Colin Crossfa138792015-04-24 17:31:52 -0700363func (c *CCBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
364 c.module.ModifyProperties(ctx)
365
Colin Cross74d1ec02015-04-28 13:30:13 -0700366 c.savedDepNames = c.module.depNames(ctx, CCDeps{})
367 c.savedDepNames.WholeStaticLibs = lastUniqueElements(c.savedDepNames.WholeStaticLibs)
368 c.savedDepNames.StaticLibs = lastUniqueElements(c.savedDepNames.StaticLibs)
369 c.savedDepNames.SharedLibs = lastUniqueElements(c.savedDepNames.SharedLibs)
370
371 staticLibs := c.savedDepNames.WholeStaticLibs
372 staticLibs = append(staticLibs, c.savedDepNames.StaticLibs...)
373 staticLibs = append(staticLibs, c.savedDepNames.LateStaticLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700374 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800375
Colin Cross74d1ec02015-04-28 13:30:13 -0700376 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, c.savedDepNames.SharedLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700377
Colin Cross74d1ec02015-04-28 13:30:13 -0700378 ret := append([]string(nil), c.savedDepNames.ObjFiles...)
379 if c.savedDepNames.CrtBegin != "" {
380 ret = append(ret, c.savedDepNames.CrtBegin)
Colin Cross21b9a242015-03-24 14:15:58 -0700381 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700382 if c.savedDepNames.CrtEnd != "" {
383 ret = append(ret, c.savedDepNames.CrtEnd)
Colin Cross21b9a242015-03-24 14:15:58 -0700384 }
385
386 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800387}
388
389// Create a ccFlags struct that collects the compile flags from global values,
390// per-target values, module type values, and per-module Blueprints properties
Colin Crossfa138792015-04-24 17:31:52 -0700391func (c *CCBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700392 flags := CCFlags{
Colin Crossfa138792015-04-24 17:31:52 -0700393 CFlags: c.Properties.Cflags,
394 CppFlags: c.Properties.Cppflags,
395 ConlyFlags: c.Properties.Conlyflags,
396 LdFlags: c.Properties.Ldflags,
397 AsFlags: c.Properties.Asflags,
398 YaccFlags: c.Properties.Yaccflags,
399 Nocrt: c.Properties.Nocrt,
Colin Cross97ba0732015-03-23 17:50:24 -0700400 Toolchain: toolchain,
Colin Crossfa138792015-04-24 17:31:52 -0700401 Clang: c.Properties.Clang,
Colin Cross3f40fa42015-01-30 17:27:36 -0800402 }
Colin Cross28344522015-04-22 13:07:53 -0700403
404 // Include dir cflags
Colin Crossfa138792015-04-24 17:31:52 -0700405 rootIncludeDirs := pathtools.PrefixPaths(c.Properties.Include_dirs, ctx.AConfig().SrcDir())
406 localIncludeDirs := pathtools.PrefixPaths(c.Properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -0700407 flags.GlobalFlags = append(flags.GlobalFlags,
408 includeDirsToFlags(rootIncludeDirs),
409 includeDirsToFlags(localIncludeDirs))
410
Colin Crossfa138792015-04-24 17:31:52 -0700411 if !c.Properties.No_default_compiler_flags {
412 if c.Properties.Sdk_version == "" || ctx.Host() {
Colin Cross28344522015-04-22 13:07:53 -0700413 flags.GlobalFlags = append(flags.GlobalFlags,
414 "${commonGlobalIncludes}",
415 toolchain.IncludeFlags(),
416 "-I${SrcDir}/libnativehelper/include/nativehelper")
417 }
418
419 flags.GlobalFlags = append(flags.GlobalFlags, []string{
420 "-I" + common.ModuleSrcDir(ctx),
421 "-I" + common.ModuleOutDir(ctx),
422 "-I" + common.ModuleGenDir(ctx),
423 }...)
424 }
425
Colin Crossfa138792015-04-24 17:31:52 -0700426 instructionSet := c.Properties.Instruction_set
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700427 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
428 if err != nil {
429 ctx.ModuleErrorf("%s", err)
430 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800431
Colin Crossaf19a292015-03-18 12:07:10 -0700432 // TODO: debug
Colin Crossfa138792015-04-24 17:31:52 -0700433 flags.CFlags = append(flags.CFlags, c.Properties.Release.Cflags...)
Colin Crossaf19a292015-03-18 12:07:10 -0700434
Colin Cross28d76592015-03-26 16:14:04 -0700435 if ctx.Host() && !ctx.ContainsProperty("clang") {
Colin Cross97ba0732015-03-23 17:50:24 -0700436 flags.Clang = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800437 }
438
Colin Cross97ba0732015-03-23 17:50:24 -0700439 if flags.Clang {
440 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
Colin Crossfa138792015-04-24 17:31:52 -0700441 flags.CFlags = append(flags.CFlags, c.Properties.Clang_cflags...)
442 flags.AsFlags = append(flags.AsFlags, c.Properties.Clang_asflags...)
Colin Cross97ba0732015-03-23 17:50:24 -0700443 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
444 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
445 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800446
Colin Cross97ba0732015-03-23 17:50:24 -0700447 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
448 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700449 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700450 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700451 }
452
Colin Cross3f40fa42015-01-30 17:27:36 -0800453 target := "-target " + toolchain.ClangTriple()
454 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
455
Colin Cross97ba0732015-03-23 17:50:24 -0700456 flags.CFlags = append(flags.CFlags, target, gccPrefix)
457 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
458 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800459
Colin Crossf6566ed2015-03-24 11:13:38 -0700460 if ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800461 gccToolchain := "--gcc-toolchain=" + toolchain.GccRoot()
462 sysroot := "--sysroot=" + filepath.Join(toolchain.GccRoot(), "sysroot")
463
464 // TODO: also need more -B, -L flags to make host builds hermetic
Colin Cross97ba0732015-03-23 17:50:24 -0700465 flags.CFlags = append(flags.CFlags, gccToolchain, sysroot)
466 flags.AsFlags = append(flags.AsFlags, gccToolchain, sysroot)
467 flags.LdFlags = append(flags.LdFlags, gccToolchain, sysroot)
Colin Cross3f40fa42015-01-30 17:27:36 -0800468 }
469 }
470
Colin Crossfa138792015-04-24 17:31:52 -0700471 if !c.Properties.No_default_compiler_flags {
472 if ctx.Device() && !c.Properties.Allow_undefined_symbols {
Colin Cross97ba0732015-03-23 17:50:24 -0700473 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800474 }
475
Colin Cross56b4d452015-04-21 17:38:44 -0700476 flags.GlobalFlags = append(flags.GlobalFlags, instructionSetFlags)
477
Colin Cross97ba0732015-03-23 17:50:24 -0700478 if flags.Clang {
479 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700480 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800481 toolchain.ClangCflags(),
482 "${commonClangGlobalCflags}",
Colin Cross56b4d452015-04-21 17:38:44 -0700483 fmt.Sprintf("${%sClangGlobalCflags}", ctx.Arch().HostOrDevice))
Colin Cross3f40fa42015-01-30 17:27:36 -0800484 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700485 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700486 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800487 toolchain.Cflags(),
488 "${commonGlobalCflags}",
Colin Cross56b4d452015-04-21 17:38:44 -0700489 fmt.Sprintf("${%sGlobalCflags}", ctx.Arch().HostOrDevice))
Colin Cross3f40fa42015-01-30 17:27:36 -0800490 }
491
Colin Crossf6566ed2015-03-24 11:13:38 -0700492 if ctx.Device() {
Colin Crossfa138792015-04-24 17:31:52 -0700493 if c.Properties.Rtti {
Colin Cross97ba0732015-03-23 17:50:24 -0700494 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800495 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700496 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800497 }
498 }
499
Colin Cross97ba0732015-03-23 17:50:24 -0700500 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800501
Colin Cross97ba0732015-03-23 17:50:24 -0700502 if flags.Clang {
503 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
504 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800505 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700506 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
507 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800508 }
Colin Cross28344522015-04-22 13:07:53 -0700509
510 if ctx.Host() {
Colin Crossfa138792015-04-24 17:31:52 -0700511 flags.LdFlags = append(flags.LdFlags, c.Properties.Host_ldlibs...)
Colin Cross28344522015-04-22 13:07:53 -0700512 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800513 }
514
Colin Cross0676e2d2015-04-24 17:39:18 -0700515 flags = c.ccModuleType().flags(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800516
517 // Optimization to reduce size of build.ninja
518 // Replace the long list of flags for each file with a module-local variable
Colin Cross97ba0732015-03-23 17:50:24 -0700519 ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " "))
520 ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " "))
521 ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " "))
522 flags.CFlags = []string{"$cflags"}
523 flags.CppFlags = []string{"$cppflags"}
524 flags.AsFlags = []string{"$asflags"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800525
526 return flags
527}
528
Colin Cross0676e2d2015-04-24 17:39:18 -0700529func (c *CCBase) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross3f40fa42015-01-30 17:27:36 -0800530 return flags
531}
532
533// Compile a list of source files into objects a specified subdirectory
Colin Crossfa138792015-04-24 17:31:52 -0700534func (c *CCBase) customCompileObjs(ctx common.AndroidModuleContext, flags CCFlags,
Colin Cross581c1892015-04-07 16:50:10 -0700535 subdir string, srcFiles []string) []string {
536
537 buildFlags := ccFlagsToBuilderFlags(flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800538
Colin Crossfce53272015-04-08 11:21:40 -0700539 srcFiles = common.ExpandSources(ctx, srcFiles)
Colin Cross581c1892015-04-07 16:50:10 -0700540 srcFiles, deps := genSources(ctx, srcFiles, buildFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800541
Colin Cross581c1892015-04-07 16:50:10 -0700542 return TransformSourceToObj(ctx, subdir, srcFiles, buildFlags, deps)
Colin Cross3f40fa42015-01-30 17:27:36 -0800543}
544
Colin Crossfa138792015-04-24 17:31:52 -0700545// Compile files listed in c.Properties.Srcs into objects
546func (c *CCBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800547
Colin Crossfa138792015-04-24 17:31:52 -0700548 if c.Properties.SkipCompileObjs {
Colin Cross3f40fa42015-01-30 17:27:36 -0800549 return nil
550 }
551
Colin Crossfa138792015-04-24 17:31:52 -0700552 return c.customCompileObjs(ctx, flags, "", c.Properties.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800553}
554
Colin Cross5049f022015-03-18 13:28:46 -0700555// Compile generated source files from dependencies
Colin Crossfa138792015-04-24 17:31:52 -0700556func (c *CCBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700557 var srcs []string
558
Colin Crossfa138792015-04-24 17:31:52 -0700559 if c.Properties.SkipCompileObjs {
Colin Cross5049f022015-03-18 13:28:46 -0700560 return nil
561 }
562
563 ctx.VisitDirectDeps(func(module blueprint.Module) {
564 if gen, ok := module.(genrule.SourceFileGenerator); ok {
565 srcs = append(srcs, gen.GeneratedSourceFiles()...)
566 }
567 })
568
569 if len(srcs) == 0 {
570 return nil
571 }
572
Colin Cross581c1892015-04-07 16:50:10 -0700573 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags), nil)
Colin Cross5049f022015-03-18 13:28:46 -0700574}
575
Colin Crossfa138792015-04-24 17:31:52 -0700576func (c *CCBase) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800577 return ""
578}
579
Colin Crossfa138792015-04-24 17:31:52 -0700580func (c *CCBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800581 names []string) (modules []common.AndroidModule,
Colin Cross28344522015-04-22 13:07:53 -0700582 outputFiles []string, exportedFlags []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800583
584 for _, n := range names {
585 found := false
586 ctx.VisitDirectDeps(func(m blueprint.Module) {
587 otherName := ctx.OtherModuleName(m)
588 if otherName != n {
589 return
590 }
591
Colin Cross97ba0732015-03-23 17:50:24 -0700592 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800593 if a.Disabled() {
594 // If a cc_library host+device module depends on a library that exists as both
595 // cc_library_shared and cc_library_host_shared, it will end up with two
596 // dependencies with the same name, one of which is marked disabled for each
597 // of host and device. Ignore the disabled one.
598 return
599 }
600 if a.HostOrDevice() != ctx.Arch().HostOrDevice {
601 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
602 otherName)
603 return
604 }
605
606 if outputFile := a.outputFile(); outputFile != "" {
607 if found {
608 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
609 return
610 }
611 outputFiles = append(outputFiles, outputFile)
612 modules = append(modules, a)
Colin Cross28344522015-04-22 13:07:53 -0700613 if i, ok := a.(ccExportedFlagsProducer); ok {
614 exportedFlags = append(exportedFlags, i.exportedFlags()...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800615 }
616 found = true
617 } else {
618 ctx.ModuleErrorf("module %q missing output file", otherName)
619 return
620 }
621 } else {
622 ctx.ModuleErrorf("module %q not an android module", otherName)
623 return
624 }
625 })
626 if !found {
627 ctx.ModuleErrorf("unsatisified dependency on %q", n)
628 }
629 }
630
Colin Cross28344522015-04-22 13:07:53 -0700631 return modules, outputFiles, exportedFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800632}
633
Colin Cross21b9a242015-03-24 14:15:58 -0700634// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
635// containing paths
Colin Crossfa138792015-04-24 17:31:52 -0700636func (c *CCBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700637 var depPaths CCDeps
Colin Cross28344522015-04-22 13:07:53 -0700638 var newCflags []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800639
Colin Cross21b9a242015-03-24 14:15:58 -0700640 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800641
Colin Cross28344522015-04-22 13:07:53 -0700642 wholeStaticLibModules, depPaths.WholeStaticLibs, newCflags =
Colin Cross21b9a242015-03-24 14:15:58 -0700643 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
Colin Cross28344522015-04-22 13:07:53 -0700644 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800645
Colin Cross21b9a242015-03-24 14:15:58 -0700646 for _, m := range wholeStaticLibModules {
647 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
648 depPaths.WholeStaticLibObjFiles =
649 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
650 } else {
651 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
652 }
653 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800654
Colin Cross28344522015-04-22 13:07:53 -0700655 _, depPaths.StaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.StaticLibs)
656 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700657
Colin Cross28344522015-04-22 13:07:53 -0700658 _, depPaths.LateStaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
659 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700660
Colin Cross28344522015-04-22 13:07:53 -0700661 _, depPaths.SharedLibs, newCflags = c.depsToPathsFromList(ctx, depNames.SharedLibs)
662 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700663
664 ctx.VisitDirectDeps(func(m blueprint.Module) {
Dan Albertc3144b12015-04-28 18:17:56 -0700665 if obj, ok := m.(ccObjectProvider); ok {
Colin Cross21b9a242015-03-24 14:15:58 -0700666 otherName := ctx.OtherModuleName(m)
667 if otherName == depNames.CrtBegin {
Colin Crossfa138792015-04-24 17:31:52 -0700668 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700669 depPaths.CrtBegin = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700670 }
671 } else if otherName == depNames.CrtEnd {
Colin Crossfa138792015-04-24 17:31:52 -0700672 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700673 depPaths.CrtEnd = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700674 }
675 } else {
Dan Albertc3144b12015-04-28 18:17:56 -0700676 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.object().outputFile())
Colin Cross21b9a242015-03-24 14:15:58 -0700677 }
678 }
679 })
680
681 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800682}
683
Colin Crossfa138792015-04-24 17:31:52 -0700684// CCLinked contains the properties and members used by libraries and executables
685type CCLinked struct {
686 CCBase
Colin Crossed4cf0b2015-03-26 14:43:45 -0700687
688 dynamicProperties struct {
Colin Cross18b6dc52015-04-28 13:20:37 -0700689 VariantIsShared bool `blueprint:"mutated"`
690 VariantIsStatic bool `blueprint:"mutated"`
691 VariantIsStaticBinary bool `blueprint:"mutated"`
Colin Crossed4cf0b2015-03-26 14:43:45 -0700692 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800693}
694
Colin Crossfa138792015-04-24 17:31:52 -0700695func newCCDynamic(dynamic *CCLinked, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700696 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
697
Colin Crossed4cf0b2015-03-26 14:43:45 -0700698 props = append(props, &dynamic.dynamicProperties)
699
Colin Crossfa138792015-04-24 17:31:52 -0700700 return newCCBase(&dynamic.CCBase, module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700701}
702
Colin Crossfa138792015-04-24 17:31:52 -0700703func (c *CCLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross28d76592015-03-26 16:14:04 -0700704 if ctx.ContainsProperty("system_shared_libs") {
Colin Crossfa138792015-04-24 17:31:52 -0700705 return c.Properties.System_shared_libs
706 } else if ctx.Device() && c.Properties.Sdk_version == "" {
Colin Cross577f6e42015-03-27 18:23:34 -0700707 return []string{"libc", "libm"}
Colin Cross28d76592015-03-26 16:14:04 -0700708 } else {
Colin Cross577f6e42015-03-27 18:23:34 -0700709 return nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800710 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800711}
712
Colin Crossfa138792015-04-24 17:31:52 -0700713func (c *CCLinked) stl(ctx common.AndroidBaseContext) string {
714 if c.Properties.Sdk_version != "" && ctx.Device() {
715 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700716 case "":
717 return "ndk_system"
718 case "c++_shared", "c++_static",
719 "stlport_shared", "stlport_static",
720 "gnustl_static":
Colin Crossfa138792015-04-24 17:31:52 -0700721 return "ndk_lib" + c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700722 default:
Colin Crossfa138792015-04-24 17:31:52 -0700723 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700724 return ""
725 }
726 }
727
Colin Crossfa138792015-04-24 17:31:52 -0700728 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700729 case "libc++", "libc++_static",
730 "stlport", "stlport_static",
731 "libstdc++":
Colin Crossfa138792015-04-24 17:31:52 -0700732 return c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700733 case "none":
734 return ""
735 case "":
Colin Cross18b6dc52015-04-28 13:20:37 -0700736 if c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700737 return "libc++_static"
Colin Cross18b6dc52015-04-28 13:20:37 -0700738 } else {
739 return "libc++" // TODO: mingw needs libstdc++
Colin Crossed4cf0b2015-03-26 14:43:45 -0700740 }
741 default:
Colin Crossfa138792015-04-24 17:31:52 -0700742 ctx.ModuleErrorf("stl: %q is not a supported STL", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700743 return ""
744 }
745}
746
Colin Cross712fc022015-04-27 11:13:34 -0700747var (
748 hostDynamicGccLibs = []string{"-lgcc_s", "-lgcc", "-lc", "-lgcc_s", "-lgcc"}
749 hostStaticGccLibs = []string{"-Wl,--start-group", "-lgcc", "-lgcc_eh", "-lc", "-Wl,--end-group"}
750)
751
Colin Crosse11befc2015-04-27 17:49:17 -0700752func (c *CCLinked) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700753 stl := c.stl(ctx)
754 if ctx.Failed() {
755 return flags
756 }
757
758 switch stl {
759 case "libc++", "libc++_static":
760 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
Colin Cross28344522015-04-22 13:07:53 -0700761 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/external/libcxx/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700762 if ctx.Host() {
763 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
764 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross712fc022015-04-27 11:13:34 -0700765 flags.LdFlags = append(flags.LdFlags, "-lm", "-lpthread")
Colin Cross18b6dc52015-04-28 13:20:37 -0700766 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700767 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700768 } else {
769 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700770 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700771 }
772 case "stlport", "stlport_static":
773 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700774 flags.CFlags = append(flags.CFlags,
775 "-I${SrcDir}/external/stlport/stlport",
776 "-I${SrcDir}/bionic/libstdc++/include",
777 "-I${SrcDir}/bionic")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700778 }
779 case "libstdc++":
780 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
781 // tree is in good enough shape to not need it.
782 // Host builds will use GNU libstdc++.
783 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700784 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/bionic/libstdc++/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700785 }
786 case "ndk_system":
Colin Cross1332b002015-04-07 17:11:30 -0700787 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources/"
Colin Cross28344522015-04-22 13:07:53 -0700788 flags.CFlags = append(flags.CFlags, "-isystem "+ndkSrcRoot+"cxx-stl/system/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700789 case "ndk_libc++_shared", "ndk_libc++_static":
790 // TODO(danalbert): This really shouldn't be here...
791 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
792 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
793 // Nothing
794 case "":
795 // None or error.
796 if ctx.Host() {
797 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
798 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross18b6dc52015-04-28 13:20:37 -0700799 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700800 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700801 } else {
802 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700803 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700804 }
805 default:
Colin Crossfa138792015-04-24 17:31:52 -0700806 panic(fmt.Errorf("Unknown stl in CCLinked.Flags: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700807 }
808
809 return flags
810}
811
Colin Crosse11befc2015-04-27 17:49:17 -0700812func (c *CCLinked) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
813 depNames = c.CCBase.depNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800814
Colin Crossed4cf0b2015-03-26 14:43:45 -0700815 stl := c.stl(ctx)
816 if ctx.Failed() {
817 return depNames
818 }
819
820 switch stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700821 case "libstdc++":
822 if ctx.Device() {
823 depNames.SharedLibs = append(depNames.SharedLibs, stl)
824 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700825 case "libc++", "libc++_static":
826 if stl == "libc++" {
827 depNames.SharedLibs = append(depNames.SharedLibs, stl)
828 } else {
829 depNames.StaticLibs = append(depNames.StaticLibs, stl)
830 }
831 if ctx.Device() {
832 if ctx.Arch().ArchType == common.Arm {
833 depNames.StaticLibs = append(depNames.StaticLibs, "libunwind_llvm")
834 }
835 if c.staticBinary() {
836 depNames.StaticLibs = append(depNames.StaticLibs, "libdl")
837 } else {
838 depNames.SharedLibs = append(depNames.SharedLibs, "libdl")
839 }
840 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700841 case "stlport":
842 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
843 case "stlport_static":
844 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
845 case "":
846 // None or error.
847 case "ndk_system":
848 // TODO: Make a system STL prebuilt for the NDK.
849 // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have
Colin Crossfa138792015-04-24 17:31:52 -0700850 // its own includes. The includes are handled in CCBase.Flags().
Colin Cross577f6e42015-03-27 18:23:34 -0700851 depNames.SharedLibs = append([]string{"libstdc++"}, depNames.SharedLibs...)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700852 case "ndk_libc++_shared", "ndk_libstlport_shared":
853 depNames.SharedLibs = append(depNames.SharedLibs, stl)
854 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
855 depNames.StaticLibs = append(depNames.StaticLibs, stl)
856 default:
Colin Crosse11befc2015-04-27 17:49:17 -0700857 panic(fmt.Errorf("Unknown stl in CCLinked.depNames: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700858 }
859
Colin Cross74d1ec02015-04-28 13:30:13 -0700860 if ctx.ModuleName() != "libcompiler_rt-extras" {
861 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
862 }
863
Colin Crossf6566ed2015-03-24 11:13:38 -0700864 if ctx.Device() {
Colin Cross77b00fa2015-03-16 16:15:49 -0700865 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700866 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700867
Colin Cross18b6dc52015-04-28 13:20:37 -0700868 if !c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700869 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
870 }
Colin Cross577f6e42015-03-27 18:23:34 -0700871
Colin Crossfa138792015-04-24 17:31:52 -0700872 if c.Properties.Sdk_version != "" {
873 version := c.Properties.Sdk_version
Colin Cross577f6e42015-03-27 18:23:34 -0700874 depNames.SharedLibs = append(depNames.SharedLibs,
875 "ndk_libc."+version,
876 "ndk_libm."+version,
877 )
878 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800879 }
880
Colin Cross21b9a242015-03-24 14:15:58 -0700881 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800882}
883
Colin Crossed4cf0b2015-03-26 14:43:45 -0700884// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
885type ccLinkedInterface interface {
886 // Returns true if the build options for the module have selected a static or shared build
887 buildStatic() bool
888 buildShared() bool
889
890 // Sets whether a specific variant is static or shared
Colin Cross18b6dc52015-04-28 13:20:37 -0700891 setStatic(bool)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700892
Colin Cross18b6dc52015-04-28 13:20:37 -0700893 // Returns whether a specific variant is a static library or binary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700894 static() bool
Colin Cross18b6dc52015-04-28 13:20:37 -0700895
896 // Returns whether a module is a static binary
897 staticBinary() bool
Colin Crossed4cf0b2015-03-26 14:43:45 -0700898}
899
900var _ ccLinkedInterface = (*CCLibrary)(nil)
901var _ ccLinkedInterface = (*CCBinary)(nil)
902
Colin Crossfa138792015-04-24 17:31:52 -0700903func (c *CCLinked) static() bool {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700904 return c.dynamicProperties.VariantIsStatic
905}
906
Colin Cross18b6dc52015-04-28 13:20:37 -0700907func (c *CCLinked) staticBinary() bool {
908 return c.dynamicProperties.VariantIsStaticBinary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700909}
910
Colin Cross18b6dc52015-04-28 13:20:37 -0700911func (c *CCLinked) setStatic(static bool) {
912 c.dynamicProperties.VariantIsStatic = static
Colin Crossed4cf0b2015-03-26 14:43:45 -0700913}
914
Colin Cross28344522015-04-22 13:07:53 -0700915type ccExportedFlagsProducer interface {
916 exportedFlags() []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800917}
918
919//
920// Combined static+shared libraries
921//
922
Colin Cross97ba0732015-03-23 17:50:24 -0700923type CCLibrary struct {
Colin Crossfa138792015-04-24 17:31:52 -0700924 CCLinked
Colin Cross3f40fa42015-01-30 17:27:36 -0800925
Colin Cross28344522015-04-22 13:07:53 -0700926 reuseFrom ccLibraryInterface
927 reuseObjFiles []string
928 objFiles []string
929 exportFlags []string
930 out string
Colin Cross3f40fa42015-01-30 17:27:36 -0800931
Colin Cross97ba0732015-03-23 17:50:24 -0700932 LibraryProperties struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800933 BuildStatic bool `blueprint:"mutated"`
934 BuildShared bool `blueprint:"mutated"`
Colin Crossed4cf0b2015-03-26 14:43:45 -0700935 Static struct {
Colin Cross2732e9a2015-04-28 13:23:52 -0700936 Srcs []string `android:"arch_variant"`
937 Cflags []string `android:"arch_variant"`
938 Whole_static_libs []string `android:"arch_variant"`
939 Static_libs []string `android:"arch_variant"`
940 Shared_libs []string `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800941 } `android:"arch_variant"`
942 Shared struct {
Colin Cross2732e9a2015-04-28 13:23:52 -0700943 Srcs []string `android:"arch_variant"`
944 Cflags []string `android:"arch_variant"`
945 Whole_static_libs []string `android:"arch_variant"`
946 Static_libs []string `android:"arch_variant"`
947 Shared_libs []string `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800948 } `android:"arch_variant"`
949 }
950}
951
Colin Crossed4cf0b2015-03-26 14:43:45 -0700952func (c *CCLibrary) buildStatic() bool {
953 return c.LibraryProperties.BuildStatic
954}
955
956func (c *CCLibrary) buildShared() bool {
957 return c.LibraryProperties.BuildShared
958}
959
Colin Cross97ba0732015-03-23 17:50:24 -0700960type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700961 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -0700962 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700963 setReuseFrom(ccLibraryInterface)
964 getReuseFrom() ccLibraryInterface
965 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -0700966 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -0700967}
968
Colin Crossed4cf0b2015-03-26 14:43:45 -0700969var _ ccLibraryInterface = (*CCLibrary)(nil)
970
Colin Cross97ba0732015-03-23 17:50:24 -0700971func (c *CCLibrary) ccLibrary() *CCLibrary {
972 return c
Colin Cross3f40fa42015-01-30 17:27:36 -0800973}
974
Colin Cross97ba0732015-03-23 17:50:24 -0700975func NewCCLibrary(library *CCLibrary, module CCModuleType,
976 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
977
Colin Crossfa138792015-04-24 17:31:52 -0700978 return newCCDynamic(&library.CCLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -0700979 &library.LibraryProperties)
980}
981
982func CCLibraryFactory() (blueprint.Module, []interface{}) {
983 module := &CCLibrary{}
984
985 module.LibraryProperties.BuildShared = true
986 module.LibraryProperties.BuildStatic = true
987
988 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
989}
990
Colin Cross0676e2d2015-04-24 17:39:18 -0700991func (c *CCLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -0700992 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Cross2732e9a2015-04-28 13:23:52 -0700993 if c.static() {
994 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Static.Whole_static_libs...)
995 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Static.Static_libs...)
996 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Static.Shared_libs...)
997 } else {
Colin Crossf6566ed2015-03-24 11:13:38 -0700998 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -0700999 if c.Properties.Sdk_version == "" {
1000 depNames.CrtBegin = "crtbegin_so"
1001 depNames.CrtEnd = "crtend_so"
1002 } else {
1003 depNames.CrtBegin = "ndk_crtbegin_so." + c.Properties.Sdk_version
1004 depNames.CrtEnd = "ndk_crtend_so." + c.Properties.Sdk_version
1005 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001006 }
Colin Cross2732e9a2015-04-28 13:23:52 -07001007 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Shared.Whole_static_libs...)
1008 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Shared.Static_libs...)
1009 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Shared.Shared_libs...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001010 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001011
Colin Cross21b9a242015-03-24 14:15:58 -07001012 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001013}
1014
Colin Cross97ba0732015-03-23 17:50:24 -07001015func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001016 return c.out
1017}
1018
Colin Crossed4cf0b2015-03-26 14:43:45 -07001019func (c *CCLibrary) getReuseObjFiles() []string {
1020 return c.reuseObjFiles
1021}
1022
1023func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
1024 c.reuseFrom = reuseFrom
1025}
1026
1027func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
1028 return c.reuseFrom
1029}
1030
Colin Cross97ba0732015-03-23 17:50:24 -07001031func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001032 return c.objFiles
1033}
1034
Colin Cross28344522015-04-22 13:07:53 -07001035func (c *CCLibrary) exportedFlags() []string {
1036 return c.exportFlags
Colin Cross3f40fa42015-01-30 17:27:36 -08001037}
1038
Colin Cross0676e2d2015-04-24 17:39:18 -07001039func (c *CCLibrary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001040 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001041
Colin Cross97ba0732015-03-23 17:50:24 -07001042 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -08001043
Colin Crossd8e780d2015-04-28 17:39:43 -07001044 if c.static() {
1045 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Static.Cflags...)
1046 } else {
1047 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Shared.Cflags...)
1048 }
1049
Colin Cross18b6dc52015-04-28 13:20:37 -07001050 if !c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001051 libName := ctx.ModuleName()
1052 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
1053 sharedFlag := "-Wl,-shared"
Colin Crossfa138792015-04-24 17:31:52 -07001054 if c.Properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001055 sharedFlag = "-shared"
1056 }
Colin Crossf6566ed2015-03-24 11:13:38 -07001057 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001058 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001059 }
Colin Cross97ba0732015-03-23 17:50:24 -07001060
1061 flags.LdFlags = append(flags.LdFlags,
1062 "-Wl,--gc-sections",
1063 sharedFlag,
1064 "-Wl,-soname,"+libName+sharedLibraryExtension,
1065 )
Colin Cross3f40fa42015-01-30 17:27:36 -08001066 }
Colin Cross97ba0732015-03-23 17:50:24 -07001067
1068 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001069}
1070
Colin Cross97ba0732015-03-23 17:50:24 -07001071func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1072 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001073
1074 staticFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001075 objFilesStatic := c.customCompileObjs(ctx, staticFlags, common.DeviceStaticLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001076 c.LibraryProperties.Static.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001077
1078 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001079 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001080
1081 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1082
1083 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1084
1085 c.objFiles = objFiles
1086 c.out = outputFile
Colin Crossfa138792015-04-24 17:31:52 -07001087 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001088 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001089
1090 ctx.CheckbuildFile(outputFile)
1091}
1092
Colin Cross97ba0732015-03-23 17:50:24 -07001093func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1094 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001095
1096 sharedFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001097 objFilesShared := c.customCompileObjs(ctx, sharedFlags, common.DeviceSharedLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001098 c.LibraryProperties.Shared.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001099
1100 objFiles = append(objFiles, objFilesShared...)
1101
1102 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1103
Colin Cross97ba0732015-03-23 17:50:24 -07001104 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001105 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, false,
Colin Cross77b00fa2015-03-16 16:15:49 -07001106 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001107
1108 c.out = outputFile
Colin Crossfa138792015-04-24 17:31:52 -07001109 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001110 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001111}
1112
Colin Cross97ba0732015-03-23 17:50:24 -07001113func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1114 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001115
1116 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001117 if c.getReuseFrom().ccLibrary() == c {
1118 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001119 } else {
Colin Cross2732e9a2015-04-28 13:23:52 -07001120 if c.getReuseFrom().ccLibrary().LibraryProperties.Static.Cflags == nil &&
1121 c.LibraryProperties.Shared.Cflags == nil {
1122 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
1123 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001124 }
1125
Colin Crossed4cf0b2015-03-26 14:43:45 -07001126 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001127 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1128 } else {
1129 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1130 }
1131}
1132
Colin Cross97ba0732015-03-23 17:50:24 -07001133func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001134 // Static libraries do not get installed.
1135}
1136
Colin Cross97ba0732015-03-23 17:50:24 -07001137func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001138 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001139 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001140 installDir = "lib64"
1141 }
1142
Colin Crossfa138792015-04-24 17:31:52 -07001143 ctx.InstallFile(filepath.Join(installDir, c.Properties.Relative_install_path), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001144}
1145
Colin Cross97ba0732015-03-23 17:50:24 -07001146func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001147 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001148 c.installStaticLibrary(ctx, flags)
1149 } else {
1150 c.installSharedLibrary(ctx, flags)
1151 }
1152}
1153
Colin Cross3f40fa42015-01-30 17:27:36 -08001154//
1155// Objects (for crt*.o)
1156//
1157
Dan Albertc3144b12015-04-28 18:17:56 -07001158type ccObjectProvider interface {
1159 object() *ccObject
1160}
1161
Colin Cross3f40fa42015-01-30 17:27:36 -08001162type ccObject struct {
Colin Crossfa138792015-04-24 17:31:52 -07001163 CCBase
Colin Cross3f40fa42015-01-30 17:27:36 -08001164 out string
1165}
1166
Dan Albertc3144b12015-04-28 18:17:56 -07001167func (c *ccObject) object() *ccObject {
1168 return c
1169}
1170
Colin Cross97ba0732015-03-23 17:50:24 -07001171func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001172 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001173
Colin Crossfa138792015-04-24 17:31:52 -07001174 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001175}
1176
1177func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1178 // object files can't have any dynamic dependencies
1179 return nil
1180}
1181
Colin Cross0676e2d2015-04-24 17:39:18 -07001182func (*ccObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001183 // object files can't have any dynamic dependencies
1184 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001185}
1186
1187func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001188 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001189
Colin Cross97ba0732015-03-23 17:50:24 -07001190 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001191
1192 var outputFile string
1193 if len(objFiles) == 1 {
1194 outputFile = objFiles[0]
1195 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001196 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+objectExtension)
Colin Cross3f40fa42015-01-30 17:27:36 -08001197 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1198 }
1199
1200 c.out = outputFile
1201
1202 ctx.CheckbuildFile(outputFile)
1203}
1204
Colin Cross97ba0732015-03-23 17:50:24 -07001205func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001206 // Object files do not get installed.
1207}
1208
Colin Cross3f40fa42015-01-30 17:27:36 -08001209func (c *ccObject) outputFile() string {
1210 return c.out
1211}
1212
Dan Albertc3144b12015-04-28 18:17:56 -07001213var _ ccObjectProvider = (*ccObject)(nil)
1214
Colin Cross3f40fa42015-01-30 17:27:36 -08001215//
1216// Executables
1217//
1218
Colin Cross97ba0732015-03-23 17:50:24 -07001219type CCBinary struct {
Colin Crossfa138792015-04-24 17:31:52 -07001220 CCLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001221 out string
Colin Crossd350ecd2015-04-28 13:25:36 -07001222 installFile string
Colin Cross97ba0732015-03-23 17:50:24 -07001223 BinaryProperties struct {
1224 // static_executable: compile executable with -static
1225 Static_executable bool
1226
1227 // stem: set the name of the output
1228 Stem string `android:"arch_variant"`
1229
Colin Cross4ae185c2015-03-26 15:12:10 -07001230 // suffix: append to the name of the output
1231 Suffix string `android:"arch_variant"`
1232
Colin Cross97ba0732015-03-23 17:50:24 -07001233 // prefix_symbols: if set, add an extra objcopy --prefix-symbols= step
1234 Prefix_symbols string
1235 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001236}
1237
Colin Crossed4cf0b2015-03-26 14:43:45 -07001238func (c *CCBinary) buildStatic() bool {
1239 return c.BinaryProperties.Static_executable
1240}
1241
1242func (c *CCBinary) buildShared() bool {
1243 return !c.BinaryProperties.Static_executable
1244}
1245
Colin Cross97ba0732015-03-23 17:50:24 -07001246func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001247 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001248 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001249 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001250 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001251
1252 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001253}
1254
Colin Cross0676e2d2015-04-24 17:39:18 -07001255func (c *CCBinary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001256 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001257 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001258 if c.Properties.Sdk_version == "" {
1259 if c.BinaryProperties.Static_executable {
1260 depNames.CrtBegin = "crtbegin_static"
1261 } else {
1262 depNames.CrtBegin = "crtbegin_dynamic"
1263 }
1264 depNames.CrtEnd = "crtend_android"
Colin Cross3f40fa42015-01-30 17:27:36 -08001265 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001266 if c.BinaryProperties.Static_executable {
1267 depNames.CrtBegin = "ndk_crtbegin_static." + c.Properties.Sdk_version
1268 } else {
1269 depNames.CrtBegin = "ndk_crtbegin_dynamic." + c.Properties.Sdk_version
1270 }
1271 depNames.CrtEnd = "ndk_crtend_android." + c.Properties.Sdk_version
Colin Cross3f40fa42015-01-30 17:27:36 -08001272 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001273
1274 if c.BinaryProperties.Static_executable {
Colin Cross74d1ec02015-04-28 13:30:13 -07001275 if c.stl(ctx) == "libc++_static" {
1276 depNames.StaticLibs = append(depNames.StaticLibs, "libm", "libc", "libdl")
1277 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001278 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1279 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1280 // move them to the beginning of deps.LateStaticLibs
1281 var groupLibs []string
1282 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1283 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1284 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1285 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001286 }
Colin Cross21b9a242015-03-24 14:15:58 -07001287 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001288}
1289
Colin Cross97ba0732015-03-23 17:50:24 -07001290func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001291 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001292
Colin Cross1f8f2342015-03-26 16:09:47 -07001293 props = append(props, &binary.BinaryProperties)
1294
Colin Crossfa138792015-04-24 17:31:52 -07001295 return newCCDynamic(&binary.CCLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001296}
1297
Colin Cross97ba0732015-03-23 17:50:24 -07001298func CCBinaryFactory() (blueprint.Module, []interface{}) {
1299 module := &CCBinary{}
1300
1301 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001302}
1303
Colin Cross18b6dc52015-04-28 13:20:37 -07001304func (c *CCBinary) ModifyProperties(ctx common.AndroidBaseContext) {
1305 if c.BinaryProperties.Static_executable {
1306 c.dynamicProperties.VariantIsStaticBinary = true
1307 }
1308}
1309
Colin Cross0676e2d2015-04-24 17:39:18 -07001310func (c *CCBinary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001311 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001312
Colin Cross97ba0732015-03-23 17:50:24 -07001313 flags.CFlags = append(flags.CFlags, "-fpie")
1314
Colin Crossf6566ed2015-03-24 11:13:38 -07001315 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001316 if c.BinaryProperties.Static_executable {
1317 // Clang driver needs -static to create static executable.
1318 // However, bionic/linker uses -shared to overwrite.
1319 // Linker for x86 targets does not allow coexistance of -static and -shared,
1320 // so we add -static only if -shared is not used.
1321 if !inList("-shared", flags.LdFlags) {
1322 flags.LdFlags = append(flags.LdFlags, "-static")
1323 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001324
Colin Crossed4cf0b2015-03-26 14:43:45 -07001325 flags.LdFlags = append(flags.LdFlags,
1326 "-nostdlib",
1327 "-Bstatic",
1328 "-Wl,--gc-sections",
1329 )
1330
1331 } else {
1332 linker := "/system/bin/linker"
1333 if flags.Toolchain.Is64Bit() {
1334 linker = "/system/bin/linker64"
1335 }
1336
1337 flags.LdFlags = append(flags.LdFlags,
1338 "-nostdlib",
1339 "-Bdynamic",
1340 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1341 "-Wl,--gc-sections",
1342 "-Wl,-z,nocopyreloc",
1343 )
1344 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001345 }
1346
Colin Cross97ba0732015-03-23 17:50:24 -07001347 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001348}
1349
Colin Cross97ba0732015-03-23 17:50:24 -07001350func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1351 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001352
Colin Crossfa138792015-04-24 17:31:52 -07001353 if !c.BinaryProperties.Static_executable && inList("libc", c.Properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001354 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1355 "from static libs or set static_executable: true")
1356 }
1357
1358 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001359 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001360 if c.BinaryProperties.Prefix_symbols != "" {
1361 afterPrefixSymbols := outputFile
1362 outputFile = outputFile + ".intermediate"
1363 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1364 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1365 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001366
Colin Cross97ba0732015-03-23 17:50:24 -07001367 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001368 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001369 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001370}
Colin Cross3f40fa42015-01-30 17:27:36 -08001371
Colin Cross97ba0732015-03-23 17:50:24 -07001372func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossd350ecd2015-04-28 13:25:36 -07001373 c.installFile = ctx.InstallFile(filepath.Join("bin", c.Properties.Relative_install_path), c.out)
1374}
1375
1376func (c *CCBinary) HostToolPath() string {
1377 if c.HostOrDevice().Host() {
1378 return c.installFile
1379 }
1380 return ""
Dan Albertc403f7c2015-03-18 14:01:18 -07001381}
1382
Colin Cross9ffb4f52015-04-24 17:48:09 -07001383type CCTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001384 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001385
Colin Cross9ffb4f52015-04-24 17:48:09 -07001386 TestProperties struct {
Colin Cross6b290692015-03-19 14:05:33 -07001387 // test_per_src: Create a separate test for each source file. Useful when there is
1388 // global state that can not be torn down and reset between each test suite.
1389 Test_per_src bool
1390 }
Dan Albertc403f7c2015-03-18 14:01:18 -07001391}
1392
Colin Cross9ffb4f52015-04-24 17:48:09 -07001393func (c *CCTest) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross0676e2d2015-04-24 17:39:18 -07001394 flags = c.CCBinary.flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001395
Colin Cross97ba0732015-03-23 17:50:24 -07001396 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001397 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001398 flags.CFlags = append(flags.CFlags, "-O0", "-g")
Colin Cross28344522015-04-22 13:07:53 -07001399 flags.LdFlags = append(flags.LdFlags, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001400 }
1401
1402 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross28344522015-04-22 13:07:53 -07001403 flags.CFlags = append(flags.CFlags,
1404 "-I"+filepath.Join(ctx.AConfig().SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001405
Colin Cross21b9a242015-03-24 14:15:58 -07001406 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001407}
1408
Colin Cross9ffb4f52015-04-24 17:48:09 -07001409func (c *CCTest) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001410 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
Colin Crossa8a93d32015-04-28 13:26:49 -07001411 depNames = c.CCBinary.depNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -07001412 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001413}
1414
Colin Cross9ffb4f52015-04-24 17:48:09 -07001415func (c *CCTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001416 if ctx.Device() {
Colin Crossa8a93d32015-04-28 13:26:49 -07001417 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001418 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001419 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001420 }
1421}
1422
Colin Cross9ffb4f52015-04-24 17:48:09 -07001423func (c *CCTest) testPerSrc() bool {
1424 return c.TestProperties.Test_per_src
Colin Cross6b290692015-03-19 14:05:33 -07001425}
1426
Colin Cross9ffb4f52015-04-24 17:48:09 -07001427func (c *CCTest) test() *CCTest {
1428 return c
1429}
1430
1431func NewCCTest(test *CCTest, module CCModuleType,
1432 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1433
1434 props = append(props, &test.TestProperties)
1435
1436 return NewCCBinary(&test.CCBinary, module, hod, props...)
1437}
1438
1439func CCTestFactory() (blueprint.Module, []interface{}) {
1440 module := &CCTest{}
1441
1442 return NewCCTest(module, module, common.HostAndDeviceSupported)
1443}
1444
1445type testPerSrc interface {
1446 test() *CCTest
1447 testPerSrc() bool
1448}
1449
1450var _ testPerSrc = (*CCTest)(nil)
1451
Colin Cross6b290692015-03-19 14:05:33 -07001452func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001453 if test, ok := mctx.Module().(testPerSrc); ok {
1454 if test.testPerSrc() {
1455 testNames := make([]string, len(test.test().Properties.Srcs))
1456 for i, src := range test.test().Properties.Srcs {
Colin Cross6b290692015-03-19 14:05:33 -07001457 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1458 }
1459 tests := mctx.CreateLocalVariations(testNames...)
Colin Cross9ffb4f52015-04-24 17:48:09 -07001460 for i, src := range test.test().Properties.Srcs {
1461 tests[i].(testPerSrc).test().Properties.Srcs = []string{src}
1462 tests[i].(testPerSrc).test().BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001463 }
1464 }
1465 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001466}
1467
1468//
1469// Static library
1470//
1471
Colin Cross97ba0732015-03-23 17:50:24 -07001472func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1473 module := &CCLibrary{}
1474 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001475
Colin Cross97ba0732015-03-23 17:50:24 -07001476 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001477}
1478
1479//
1480// Shared libraries
1481//
1482
Colin Cross97ba0732015-03-23 17:50:24 -07001483func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1484 module := &CCLibrary{}
1485 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001486
Colin Cross97ba0732015-03-23 17:50:24 -07001487 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001488}
1489
1490//
1491// Host static library
1492//
1493
Colin Cross97ba0732015-03-23 17:50:24 -07001494func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1495 module := &CCLibrary{}
1496 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001497
Colin Cross97ba0732015-03-23 17:50:24 -07001498 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001499}
1500
1501//
1502// Host Shared libraries
1503//
1504
Colin Cross97ba0732015-03-23 17:50:24 -07001505func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1506 module := &CCLibrary{}
1507 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001508
Colin Cross97ba0732015-03-23 17:50:24 -07001509 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001510}
1511
1512//
1513// Host Binaries
1514//
1515
Colin Cross97ba0732015-03-23 17:50:24 -07001516func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1517 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001518
Colin Cross97ba0732015-03-23 17:50:24 -07001519 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001520}
1521
1522//
Colin Cross1f8f2342015-03-26 16:09:47 -07001523// Host Tests
1524//
1525
1526func CCTestHostFactory() (blueprint.Module, []interface{}) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001527 module := &CCTest{}
Colin Cross1f8f2342015-03-26 16:09:47 -07001528 return NewCCBinary(&module.CCBinary, module, common.HostSupported,
Colin Cross9ffb4f52015-04-24 17:48:09 -07001529 &module.TestProperties)
Colin Cross1f8f2342015-03-26 16:09:47 -07001530}
1531
1532//
Colin Cross3f40fa42015-01-30 17:27:36 -08001533// Device libraries shipped with gcc
1534//
1535
1536type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001537 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001538}
1539
1540func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1541 // toolchain libraries can't have any dependencies
1542 return nil
1543}
1544
Colin Cross0676e2d2015-04-24 17:39:18 -07001545func (*toolchainLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001546 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001547 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001548}
1549
Colin Cross97ba0732015-03-23 17:50:24 -07001550func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001551 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001552
Colin Cross97ba0732015-03-23 17:50:24 -07001553 module.LibraryProperties.BuildStatic = true
1554
Colin Crossfa138792015-04-24 17:31:52 -07001555 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth,
Colin Cross21b9a242015-03-24 14:15:58 -07001556 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001557}
1558
1559func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001560 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001561
1562 libName := ctx.ModuleName() + staticLibraryExtension
1563 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1564
1565 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1566
1567 c.out = outputFile
1568
1569 ctx.CheckbuildFile(outputFile)
1570}
1571
Colin Cross97ba0732015-03-23 17:50:24 -07001572func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001573 // Toolchain libraries do not get installed.
1574}
1575
Dan Albertbe961682015-03-18 23:38:50 -07001576// NDK prebuilt libraries.
1577//
1578// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1579// either (with the exception of the shared STLs, which are installed to the app's directory rather
1580// than to the system image).
1581
1582func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1583 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
Colin Cross1332b002015-04-07 17:11:30 -07001584 ctx.AConfig().SrcDir(), version, toolchain.Name())
Dan Albertbe961682015-03-18 23:38:50 -07001585}
1586
Dan Albertc3144b12015-04-28 18:17:56 -07001587func ndkPrebuiltModuleToPath(ctx common.AndroidModuleContext, toolchain Toolchain,
1588 ext string, version string) string {
1589
1590 // NDK prebuilts are named like: ndk_NAME.EXT.SDK_VERSION.
1591 // We want to translate to just NAME.EXT
1592 name := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1593 dir := getNdkLibDir(ctx, toolchain, version)
1594 return filepath.Join(dir, name+ext)
1595}
1596
1597type ndkPrebuiltObject struct {
1598 ccObject
1599}
1600
1601func (*ndkPrebuiltObject) AndroidDynamicDependencies(
1602 ctx common.AndroidDynamicDependerModuleContext) []string {
1603
1604 // NDK objects can't have any dependencies
1605 return nil
1606}
1607
1608func (*ndkPrebuiltObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1609 // NDK objects can't have any dependencies
1610 return CCDeps{}
1611}
1612
1613func NdkPrebuiltObjectFactory() (blueprint.Module, []interface{}) {
1614 module := &ndkPrebuiltObject{}
1615 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
1616}
1617
1618func (c *ndkPrebuiltObject) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1619 deps CCDeps, objFiles []string) {
1620 // A null build step, but it sets up the output path.
1621 if !strings.HasPrefix(ctx.ModuleName(), "ndk_crt") {
1622 ctx.ModuleErrorf("NDK prebuilts must have an ndk_crt prefixed name")
1623 }
1624
1625 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, objectExtension, c.Properties.Sdk_version)
1626}
1627
1628func (c *ndkPrebuiltObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1629 // Objects do not get installed.
1630}
1631
1632var _ ccObjectProvider = (*ndkPrebuiltObject)(nil)
1633
Dan Albertbe961682015-03-18 23:38:50 -07001634type ndkPrebuiltLibrary struct {
1635 CCLibrary
1636}
1637
1638func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1639 ctx common.AndroidDynamicDependerModuleContext) []string {
1640
1641 // NDK libraries can't have any dependencies
1642 return nil
1643}
1644
Colin Cross0676e2d2015-04-24 17:39:18 -07001645func (*ndkPrebuiltLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Dan Albertbe961682015-03-18 23:38:50 -07001646 // NDK libraries can't have any dependencies
1647 return CCDeps{}
1648}
1649
1650func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1651 module := &ndkPrebuiltLibrary{}
1652 module.LibraryProperties.BuildShared = true
1653 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1654}
1655
1656func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1657 deps CCDeps, objFiles []string) {
1658 // A null build step, but it sets up the output path.
1659 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1660 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1661 }
1662
Colin Crossfa138792015-04-24 17:31:52 -07001663 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001664 c.exportFlags = []string{common.JoinWithPrefix(includeDirs, "-isystem ")}
Dan Albertbe961682015-03-18 23:38:50 -07001665
Dan Albertc3144b12015-04-28 18:17:56 -07001666 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, sharedLibraryExtension,
1667 c.Properties.Sdk_version)
Dan Albertbe961682015-03-18 23:38:50 -07001668}
1669
1670func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc3144b12015-04-28 18:17:56 -07001671 // NDK prebuilt libraries do not get installed.
Dan Albertbe961682015-03-18 23:38:50 -07001672}
1673
1674// The NDK STLs are slightly different from the prebuilt system libraries:
1675// * Are not specific to each platform version.
1676// * The libraries are not in a predictable location for each STL.
1677
1678type ndkPrebuiltStl struct {
1679 ndkPrebuiltLibrary
1680}
1681
1682type ndkPrebuiltStaticStl struct {
1683 ndkPrebuiltStl
1684}
1685
1686type ndkPrebuiltSharedStl struct {
1687 ndkPrebuiltStl
1688}
1689
1690func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1691 module := &ndkPrebuiltSharedStl{}
1692 module.LibraryProperties.BuildShared = true
1693 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1694}
1695
1696func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1697 module := &ndkPrebuiltStaticStl{}
1698 module.LibraryProperties.BuildStatic = true
1699 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1700}
1701
1702func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1703 gccVersion := toolchain.GccVersion()
1704 var libDir string
1705 switch stl {
1706 case "libstlport":
1707 libDir = "cxx-stl/stlport/libs"
1708 case "libc++":
1709 libDir = "cxx-stl/llvm-libc++/libs"
1710 case "libgnustl":
1711 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1712 }
1713
1714 if libDir != "" {
Colin Cross1332b002015-04-07 17:11:30 -07001715 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources"
Dan Albertbe961682015-03-18 23:38:50 -07001716 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1717 }
1718
1719 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1720 return ""
1721}
1722
1723func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1724 deps CCDeps, objFiles []string) {
1725 // A null build step, but it sets up the output path.
1726 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1727 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1728 }
1729
Colin Crossfa138792015-04-24 17:31:52 -07001730 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001731 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Dan Albertbe961682015-03-18 23:38:50 -07001732
1733 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1734 libExt := sharedLibraryExtension
1735 if c.LibraryProperties.BuildStatic {
1736 libExt = staticLibraryExtension
1737 }
1738
1739 stlName := strings.TrimSuffix(libName, "_shared")
1740 stlName = strings.TrimSuffix(stlName, "_static")
1741 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1742 c.out = libDir + "/" + libName + libExt
1743}
1744
Colin Cross3f40fa42015-01-30 17:27:36 -08001745func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001746 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001747 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001748 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001749 modules = mctx.CreateLocalVariations("static", "shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001750 modules[0].(ccLinkedInterface).setStatic(true)
1751 modules[1].(ccLinkedInterface).setStatic(false)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001752 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001753 modules = mctx.CreateLocalVariations("static")
Colin Cross18b6dc52015-04-28 13:20:37 -07001754 modules[0].(ccLinkedInterface).setStatic(true)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001755 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001756 modules = mctx.CreateLocalVariations("shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001757 modules[0].(ccLinkedInterface).setStatic(false)
Colin Cross3f40fa42015-01-30 17:27:36 -08001758 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001759 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001760 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001761
1762 if _, ok := c.(ccLibraryInterface); ok {
1763 reuseFrom := modules[0].(ccLibraryInterface)
1764 for _, m := range modules {
1765 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001766 }
1767 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001768 }
1769}
Colin Cross74d1ec02015-04-28 13:30:13 -07001770
1771// lastUniqueElements returns all unique elements of a slice, keeping the last copy of each
1772// modifies the slice contents in place, and returns a subslice of the original slice
1773func lastUniqueElements(list []string) []string {
1774 totalSkip := 0
1775 for i := len(list) - 1; i >= totalSkip; i-- {
1776 skip := 0
1777 for j := i - 1; j >= totalSkip; j-- {
1778 if list[i] == list[j] {
1779 skip++
1780 } else {
1781 list[j+skip] = list[j]
1782 }
1783 }
1784 totalSkip += skip
1785 }
1786 return list[totalSkip:]
1787}