blob: a5af73e116a4f18d469843d56123fe14cc75a93f [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 Cross97ba0732015-03-23 17:50:24 -0700107// ccProperties describes properties used to compile all C or C++ modules
Colin Cross3f40fa42015-01-30 17:27:36 -0800108type ccProperties struct {
109 // srcs: list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
110 Srcs []string `android:"arch_variant,arch_subtract"`
111
112 // cflags: list of module-specific flags that will be used for C and C++ compiles.
113 Cflags []string `android:"arch_variant"`
114
115 // cppflags: list of module-specific flags that will be used for C++ compiles
116 Cppflags []string `android:"arch_variant"`
117
118 // conlyflags: list of module-specific flags that will be used for C compiles
119 Conlyflags []string `android:"arch_variant"`
120
121 // asflags: list of module-specific flags that will be used for .S compiles
122 Asflags []string `android:"arch_variant"`
123
Colin Cross581c1892015-04-07 16:50:10 -0700124 // yaccflags: list of module-specific flags that will be used for .y and .yy compiles
125 Yaccflags []string
126
Colin Cross3f40fa42015-01-30 17:27:36 -0800127 // ldflags: list of module-specific flags that will be used for all link steps
128 Ldflags []string `android:"arch_variant"`
129
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700130 // instruction_set: the instruction set architecture to use to compile the C/C++
131 // module.
132 Instruction_set string `android:"arch_variant"`
133
Colin Cross3f40fa42015-01-30 17:27:36 -0800134 // include_dirs: list of directories relative to the root of the source tree that will
135 // be added to the include path using -I.
136 // If possible, don't use this. If adding paths from the current directory use
137 // local_include_dirs, if adding paths from other modules use export_include_dirs in
138 // that module.
139 Include_dirs []string `android:"arch_variant"`
140
141 // local_include_dirs: list of directories relative to the Blueprints file that will
142 // be added to the include path using -I
143 Local_include_dirs []string `android:"arch_variant"`
144
145 // export_include_dirs: list of directories relative to the Blueprints file that will
146 // be added to the include path using -I for any module that links against this module
Dan Albertbe961682015-03-18 23:38:50 -0700147 Export_include_dirs []string `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800148
149 // clang_cflags: list of module-specific flags that will be used for C and C++ compiles when
150 // compiling with clang
151 Clang_cflags []string `android:"arch_variant"`
152
153 // clang_asflags: list of module-specific flags that will be used for .S compiles when
154 // compiling with clang
155 Clang_asflags []string `android:"arch_variant"`
156
157 // system_shared_libs: list of system libraries that will be dynamically linked to
158 // shared library and executable modules. If unset, generally defaults to libc
159 // and libm. Set to [] to prevent linking against libc and libm.
160 System_shared_libs []string
161
162 // whole_static_libs: list of modules whose object files should be linked into this module
163 // in their entirety. For static library modules, all of the .o files from the intermediate
164 // directory of the dependency will be linked into this modules .a file. For a shared library,
165 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
166 Whole_static_libs []string `android:"arch_variant"`
167
168 // static_libs: list of modules that should be statically linked into this module.
169 Static_libs []string `android:"arch_variant"`
170
171 // shared_libs: list of modules that should be dynamically linked into this module.
172 Shared_libs []string `android:"arch_variant"`
173
174 // allow_undefined_symbols: allow the module to contain undefined symbols. By default,
175 // modules cannot contain undefined symbols that are not satisified by their immediate
176 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
177 // This flag should only be necessary for compiling low-level libraries like libc.
178 Allow_undefined_symbols bool
179
180 // nocrt: don't link in crt_begin and crt_end. This flag should only be necessary for
181 // compiling crt or libc.
182 Nocrt bool `android:"arch_variant"`
183
184 // no_default_compiler_flags: don't insert default compiler flags into asflags, cflags,
185 // cppflags, conlyflags, ldflags, or include_dirs
186 No_default_compiler_flags bool
187
188 // clang: compile module with clang instead of gcc
189 Clang bool `android:"arch_variant"`
190
191 // rtti: pass -frtti instead of -fno-rtti
192 Rtti bool
193
194 // host_ldlibs: -l arguments to pass to linker for host-provided shared libraries
195 Host_ldlibs []string `android:"arch_variant"`
196
197 // stl: select the STL library to use. Possible values are "libc++", "libc++_static",
198 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
199 // default
200 Stl string
201
202 // Set for combined shared/static libraries to prevent compiling object files a second time
203 SkipCompileObjs bool `blueprint:"mutated"`
Colin Crossaf19a292015-03-18 12:07:10 -0700204
205 Debug struct {
206 Cflags []string `android:"arch_variant"`
207 } `android:"arch_variant"`
208 Release struct {
209 Cflags []string `android:"arch_variant"`
210 } `android:"arch_variant"`
Colin Crossefd8e482015-03-18 17:17:35 -0700211
212 // Minimum sdk version supported when compiling against the ndk
213 Sdk_version string
Colin Cross3f40fa42015-01-30 17:27:36 -0800214}
215
216type unusedProperties struct {
217 Asan bool
218 Native_coverage bool
219 Strip string
220 Tags []string
221 Required []string
222}
223
224// Building C/C++ code is handled by objects that satisfy this interface via composition
Colin Cross97ba0732015-03-23 17:50:24 -0700225type CCModuleType interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800226 common.AndroidModule
227
Colin Cross21b9a242015-03-24 14:15:58 -0700228 // Modify the ccFlags
229 Flags(common.AndroidModuleContext, CCFlags) CCFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800230
Colin Cross21b9a242015-03-24 14:15:58 -0700231 // Return list of dependency names for use in AndroidDynamicDependencies and in depsToPaths
232 DepNames(common.AndroidBaseContext, CCDeps) CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800233
234 // Compile objects into final module
Colin Cross97ba0732015-03-23 17:50:24 -0700235 compileModule(common.AndroidModuleContext, CCFlags, CCDeps, []string)
Colin Cross3f40fa42015-01-30 17:27:36 -0800236
Dan Albertc403f7c2015-03-18 14:01:18 -0700237 // Install the built module.
Colin Cross97ba0732015-03-23 17:50:24 -0700238 installModule(common.AndroidModuleContext, CCFlags)
Dan Albertc403f7c2015-03-18 14:01:18 -0700239
Colin Cross3f40fa42015-01-30 17:27:36 -0800240 // Return the output file (.o, .a or .so) for use by other modules
241 outputFile() string
242}
243
Colin Cross97ba0732015-03-23 17:50:24 -0700244type CCDeps struct {
245 StaticLibs, SharedLibs, LateStaticLibs, WholeStaticLibs, ObjFiles, IncludeDirs []string
Colin Crossc472d572015-03-17 15:06:21 -0700246
Colin Cross21b9a242015-03-24 14:15:58 -0700247 WholeStaticLibObjFiles []string
248
Colin Cross97ba0732015-03-23 17:50:24 -0700249 CrtBegin, CrtEnd string
Colin Crossc472d572015-03-17 15:06:21 -0700250}
251
Colin Cross97ba0732015-03-23 17:50:24 -0700252type CCFlags struct {
253 GlobalFlags []string
254 AsFlags []string
255 CFlags []string
256 ConlyFlags []string
257 CppFlags []string
Colin Cross581c1892015-04-07 16:50:10 -0700258 YaccFlags []string
Colin Cross97ba0732015-03-23 17:50:24 -0700259 LdFlags []string
260 LdLibs []string
261 IncludeDirs []string
262 Nocrt bool
263 Toolchain Toolchain
264 Clang bool
Colin Crossc472d572015-03-17 15:06:21 -0700265}
266
267// ccBase contains the properties and members used by all C/C++ module types, and implements
268// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
269// and uses a ccModuleType interface to that struct to create the build steps.
270type ccBase struct {
271 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700272 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700273
274 properties ccProperties
275 unused unusedProperties
276
277 installPath string
278}
279
Colin Cross97ba0732015-03-23 17:50:24 -0700280func newCCBase(base *ccBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700281 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
282
283 base.module = module
284
285 props = append(props, &base.properties, &base.unused)
286
Colin Cross5049f022015-03-18 13:28:46 -0700287 return common.InitAndroidArchModule(module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700288}
289
Colin Cross3f40fa42015-01-30 17:27:36 -0800290func (c *ccBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
291 toolchain := c.findToolchain(ctx)
292 if ctx.Failed() {
293 return
294 }
295
Colin Cross21b9a242015-03-24 14:15:58 -0700296 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800297 if ctx.Failed() {
298 return
299 }
300
Colin Cross21b9a242015-03-24 14:15:58 -0700301 depNames := c.module.DepNames(ctx, CCDeps{})
Colin Cross3f40fa42015-01-30 17:27:36 -0800302 if ctx.Failed() {
303 return
304 }
305
Colin Cross21b9a242015-03-24 14:15:58 -0700306 deps := c.depsToPaths(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800307 if ctx.Failed() {
308 return
309 }
310
Colin Cross97ba0732015-03-23 17:50:24 -0700311 flags.IncludeDirs = append(flags.IncludeDirs, deps.IncludeDirs...)
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 Cross97ba0732015-03-23 17:50:24 -0700336func (c *ccBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800337 return c.module
338}
339
340var _ common.AndroidDynamicDepender = (*ccBase)(nil)
341
Colin Cross97ba0732015-03-23 17:50:24 -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 Cross21b9a242015-03-24 14:15:58 -0700352func (c *ccBase) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
353 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.properties.Whole_static_libs...)
354 depNames.StaticLibs = append(depNames.StaticLibs, c.properties.Static_libs...)
355 depNames.SharedLibs = append(depNames.SharedLibs, c.properties.Shared_libs...)
356
Colin Cross21b9a242015-03-24 14:15:58 -0700357 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800358}
359
360func (c *ccBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
Colin Cross21b9a242015-03-24 14:15:58 -0700361 depNames := CCDeps{}
362 depNames = c.module.DepNames(ctx, depNames)
363 staticLibs := depNames.WholeStaticLibs
364 staticLibs = append(staticLibs, depNames.StaticLibs...)
365 staticLibs = append(staticLibs, depNames.LateStaticLibs...)
366 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800367
Colin Cross21b9a242015-03-24 14:15:58 -0700368 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, depNames.SharedLibs...)
369
370 ret := append([]string(nil), depNames.ObjFiles...)
371 if depNames.CrtBegin != "" {
372 ret = append(ret, depNames.CrtBegin)
373 }
374 if depNames.CrtEnd != "" {
375 ret = append(ret, depNames.CrtEnd)
376 }
377
378 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800379}
380
381// Create a ccFlags struct that collects the compile flags from global values,
382// per-target values, module type values, and per-module Blueprints properties
Colin Cross21b9a242015-03-24 14:15:58 -0700383func (c *ccBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700384 flags := CCFlags{
385 CFlags: c.properties.Cflags,
386 CppFlags: c.properties.Cppflags,
387 ConlyFlags: c.properties.Conlyflags,
388 LdFlags: c.properties.Ldflags,
389 AsFlags: c.properties.Asflags,
Colin Cross581c1892015-04-07 16:50:10 -0700390 YaccFlags: c.properties.Yaccflags,
Colin Cross97ba0732015-03-23 17:50:24 -0700391 Nocrt: c.properties.Nocrt,
392 Toolchain: toolchain,
393 Clang: c.properties.Clang,
Colin Cross3f40fa42015-01-30 17:27:36 -0800394 }
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700395 instructionSet := c.properties.Instruction_set
396 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
397 if err != nil {
398 ctx.ModuleErrorf("%s", err)
399 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800400
Colin Crossaf19a292015-03-18 12:07:10 -0700401 // TODO: debug
Colin Cross97ba0732015-03-23 17:50:24 -0700402 flags.CFlags = append(flags.CFlags, c.properties.Release.Cflags...)
Colin Crossaf19a292015-03-18 12:07:10 -0700403
Colin Cross28d76592015-03-26 16:14:04 -0700404 if ctx.Host() && !ctx.ContainsProperty("clang") {
Colin Cross97ba0732015-03-23 17:50:24 -0700405 flags.Clang = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800406 }
407
Colin Cross97ba0732015-03-23 17:50:24 -0700408 if flags.Clang {
409 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
410 flags.CFlags = append(flags.CFlags, c.properties.Clang_cflags...)
411 flags.AsFlags = append(flags.AsFlags, c.properties.Clang_asflags...)
412 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
413 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
414 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800415
Colin Cross97ba0732015-03-23 17:50:24 -0700416 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
417 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700418 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700419 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700420 }
421
Colin Cross3f40fa42015-01-30 17:27:36 -0800422 target := "-target " + toolchain.ClangTriple()
423 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
424
Colin Cross97ba0732015-03-23 17:50:24 -0700425 flags.CFlags = append(flags.CFlags, target, gccPrefix)
426 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
427 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800428
Colin Crossf6566ed2015-03-24 11:13:38 -0700429 if ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800430 gccToolchain := "--gcc-toolchain=" + toolchain.GccRoot()
431 sysroot := "--sysroot=" + filepath.Join(toolchain.GccRoot(), "sysroot")
432
433 // TODO: also need more -B, -L flags to make host builds hermetic
Colin Cross97ba0732015-03-23 17:50:24 -0700434 flags.CFlags = append(flags.CFlags, gccToolchain, sysroot)
435 flags.AsFlags = append(flags.AsFlags, gccToolchain, sysroot)
436 flags.LdFlags = append(flags.LdFlags, gccToolchain, sysroot)
Colin Cross3f40fa42015-01-30 17:27:36 -0800437 }
438 }
439
Colin Cross1332b002015-04-07 17:11:30 -0700440 flags.IncludeDirs = pathtools.PrefixPaths(c.properties.Include_dirs, ctx.AConfig().SrcDir())
Colin Cross3f40fa42015-01-30 17:27:36 -0800441 localIncludeDirs := pathtools.PrefixPaths(c.properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross97ba0732015-03-23 17:50:24 -0700442 flags.IncludeDirs = append(flags.IncludeDirs, localIncludeDirs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800443
444 if !c.properties.No_default_compiler_flags {
Colin Cross97ba0732015-03-23 17:50:24 -0700445 flags.IncludeDirs = append(flags.IncludeDirs, []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800446 common.ModuleSrcDir(ctx),
447 common.ModuleOutDir(ctx),
448 common.ModuleGenDir(ctx),
449 }...)
450
Colin Crossefd8e482015-03-18 17:17:35 -0700451 if c.properties.Sdk_version == "" {
Colin Cross97ba0732015-03-23 17:50:24 -0700452 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/libnativehelper/include/nativehelper")
Colin Crossefd8e482015-03-18 17:17:35 -0700453 }
454
Colin Crossf6566ed2015-03-24 11:13:38 -0700455 if ctx.Device() && !c.properties.Allow_undefined_symbols {
Colin Cross97ba0732015-03-23 17:50:24 -0700456 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800457 }
458
Colin Cross97ba0732015-03-23 17:50:24 -0700459 if flags.Clang {
460 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
461 flags.GlobalFlags = []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800462 "${commonGlobalIncludes}",
463 toolchain.IncludeFlags(),
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700464 instructionSetFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800465 toolchain.ClangCflags(),
466 "${commonClangGlobalCflags}",
Colin Crossf6566ed2015-03-24 11:13:38 -0700467 fmt.Sprintf("${%sClangGlobalCflags}", ctx.Arch().HostOrDevice),
Colin Cross3f40fa42015-01-30 17:27:36 -0800468 }
469 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700470 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
471 flags.GlobalFlags = []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800472 "${commonGlobalIncludes}",
473 toolchain.IncludeFlags(),
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700474 instructionSetFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800475 toolchain.Cflags(),
476 "${commonGlobalCflags}",
Colin Crossf6566ed2015-03-24 11:13:38 -0700477 fmt.Sprintf("${%sGlobalCflags}", ctx.Arch().HostOrDevice),
Colin Cross3f40fa42015-01-30 17:27:36 -0800478 }
479 }
480
Colin Crossf6566ed2015-03-24 11:13:38 -0700481 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -0700482 flags.LdFlags = append(flags.LdFlags, c.properties.Host_ldlibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800483 }
484
Colin Crossf6566ed2015-03-24 11:13:38 -0700485 if ctx.Device() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800486 if c.properties.Rtti {
Colin Cross97ba0732015-03-23 17:50:24 -0700487 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800488 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700489 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800490 }
491 }
492
Colin Cross97ba0732015-03-23 17:50:24 -0700493 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800494
Colin Cross97ba0732015-03-23 17:50:24 -0700495 if flags.Clang {
496 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
497 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800498 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700499 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
500 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800501 }
502 }
503
Colin Cross21b9a242015-03-24 14:15:58 -0700504 flags = c.ccModuleType().Flags(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800505
506 // Optimization to reduce size of build.ninja
507 // Replace the long list of flags for each file with a module-local variable
Colin Cross97ba0732015-03-23 17:50:24 -0700508 ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " "))
509 ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " "))
510 ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " "))
511 flags.CFlags = []string{"$cflags"}
512 flags.CppFlags = []string{"$cppflags"}
513 flags.AsFlags = []string{"$asflags"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800514
515 return flags
516}
517
Colin Cross21b9a242015-03-24 14:15:58 -0700518func (c *ccBase) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross3f40fa42015-01-30 17:27:36 -0800519 return flags
520}
521
522// Compile a list of source files into objects a specified subdirectory
Colin Cross97ba0732015-03-23 17:50:24 -0700523func (c *ccBase) customCompileObjs(ctx common.AndroidModuleContext, flags CCFlags,
Colin Cross581c1892015-04-07 16:50:10 -0700524 subdir string, srcFiles []string) []string {
525
526 buildFlags := ccFlagsToBuilderFlags(flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800527
528 srcFiles = pathtools.PrefixPaths(srcFiles, common.ModuleSrcDir(ctx))
529 srcFiles = common.ExpandGlobs(ctx, srcFiles)
Colin Cross581c1892015-04-07 16:50:10 -0700530 srcFiles, deps := genSources(ctx, srcFiles, buildFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800531
Colin Cross581c1892015-04-07 16:50:10 -0700532 return TransformSourceToObj(ctx, subdir, srcFiles, buildFlags, deps)
Colin Cross3f40fa42015-01-30 17:27:36 -0800533}
534
535// Compile files listed in c.properties.Srcs into objects
Colin Cross581c1892015-04-07 16:50:10 -0700536func (c *ccBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800537
538 if c.properties.SkipCompileObjs {
539 return nil
540 }
541
Colin Cross581c1892015-04-07 16:50:10 -0700542 return c.customCompileObjs(ctx, flags, "", c.properties.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800543}
544
Colin Cross5049f022015-03-18 13:28:46 -0700545// Compile generated source files from dependencies
Colin Cross581c1892015-04-07 16:50:10 -0700546func (c *ccBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700547 var srcs []string
548
549 if c.properties.SkipCompileObjs {
550 return nil
551 }
552
553 ctx.VisitDirectDeps(func(module blueprint.Module) {
554 if gen, ok := module.(genrule.SourceFileGenerator); ok {
555 srcs = append(srcs, gen.GeneratedSourceFiles()...)
556 }
557 })
558
559 if len(srcs) == 0 {
560 return nil
561 }
562
Colin Cross581c1892015-04-07 16:50:10 -0700563 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags), nil)
Colin Cross5049f022015-03-18 13:28:46 -0700564}
565
Colin Cross3f40fa42015-01-30 17:27:36 -0800566func (c *ccBase) outputFile() string {
567 return ""
568}
569
Colin Cross21b9a242015-03-24 14:15:58 -0700570func (c *ccBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800571 names []string) (modules []common.AndroidModule,
572 outputFiles []string, exportedIncludeDirs []string) {
573
574 for _, n := range names {
575 found := false
576 ctx.VisitDirectDeps(func(m blueprint.Module) {
577 otherName := ctx.OtherModuleName(m)
578 if otherName != n {
579 return
580 }
581
Colin Cross97ba0732015-03-23 17:50:24 -0700582 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800583 if a.Disabled() {
584 // If a cc_library host+device module depends on a library that exists as both
585 // cc_library_shared and cc_library_host_shared, it will end up with two
586 // dependencies with the same name, one of which is marked disabled for each
587 // of host and device. Ignore the disabled one.
588 return
589 }
590 if a.HostOrDevice() != ctx.Arch().HostOrDevice {
591 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
592 otherName)
593 return
594 }
595
596 if outputFile := a.outputFile(); outputFile != "" {
597 if found {
598 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
599 return
600 }
601 outputFiles = append(outputFiles, outputFile)
602 modules = append(modules, a)
603 if i, ok := a.(ccExportedIncludeDirsProducer); ok {
604 exportedIncludeDirs = append(exportedIncludeDirs, i.exportedIncludeDirs()...)
605 }
606 found = true
607 } else {
608 ctx.ModuleErrorf("module %q missing output file", otherName)
609 return
610 }
611 } else {
612 ctx.ModuleErrorf("module %q not an android module", otherName)
613 return
614 }
615 })
616 if !found {
617 ctx.ModuleErrorf("unsatisified dependency on %q", n)
618 }
619 }
620
621 return modules, outputFiles, exportedIncludeDirs
622}
623
Colin Cross21b9a242015-03-24 14:15:58 -0700624// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
625// containing paths
626func (c *ccBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
627 var depPaths CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800628 var newIncludeDirs []string
629
Colin Cross21b9a242015-03-24 14:15:58 -0700630 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800631
Colin Cross21b9a242015-03-24 14:15:58 -0700632 wholeStaticLibModules, depPaths.WholeStaticLibs, newIncludeDirs =
633 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
634 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800635
Colin Cross21b9a242015-03-24 14:15:58 -0700636 for _, m := range wholeStaticLibModules {
637 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
638 depPaths.WholeStaticLibObjFiles =
639 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
640 } else {
641 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
642 }
643 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800644
Colin Cross21b9a242015-03-24 14:15:58 -0700645 _, depPaths.StaticLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.StaticLibs)
646 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
647
648 _, depPaths.LateStaticLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
649 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
650
651 _, depPaths.SharedLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.SharedLibs)
652 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
653
654 ctx.VisitDirectDeps(func(m blueprint.Module) {
655 if obj, ok := m.(*ccObject); ok {
656 otherName := ctx.OtherModuleName(m)
657 if otherName == depNames.CrtBegin {
658 if !c.properties.Nocrt {
659 depPaths.CrtBegin = obj.outputFile()
660 }
661 } else if otherName == depNames.CrtEnd {
662 if !c.properties.Nocrt {
663 depPaths.CrtEnd = obj.outputFile()
664 }
665 } else {
666 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.outputFile())
667 }
668 }
669 })
670
671 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800672}
673
Colin Crossed4cf0b2015-03-26 14:43:45 -0700674// ccLinked contains the properties and members used by libraries and executables
675type ccLinked struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800676 ccBase
Colin Crossed4cf0b2015-03-26 14:43:45 -0700677
678 dynamicProperties struct {
679 VariantIsShared bool `blueprint:"mutated"`
680 VariantIsStatic bool `blueprint:"mutated"`
681 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800682}
683
Colin Crossed4cf0b2015-03-26 14:43:45 -0700684func newCCDynamic(dynamic *ccLinked, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700685 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
686
Colin Crossed4cf0b2015-03-26 14:43:45 -0700687 props = append(props, &dynamic.dynamicProperties)
688
Colin Crossc472d572015-03-17 15:06:21 -0700689 return newCCBase(&dynamic.ccBase, module, hod, multilib, props...)
690}
691
Colin Crossed4cf0b2015-03-26 14:43:45 -0700692func (c *ccLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross28d76592015-03-26 16:14:04 -0700693 if ctx.ContainsProperty("system_shared_libs") {
694 return c.properties.System_shared_libs
Colin Cross577f6e42015-03-27 18:23:34 -0700695 } else if ctx.Device() && c.properties.Sdk_version == "" {
696 return []string{"libc", "libm"}
Colin Cross28d76592015-03-26 16:14:04 -0700697 } else {
Colin Cross577f6e42015-03-27 18:23:34 -0700698 return nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800699 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800700}
701
Colin Crossed4cf0b2015-03-26 14:43:45 -0700702func (c *ccLinked) stl(ctx common.AndroidBaseContext) string {
Colin Cross577f6e42015-03-27 18:23:34 -0700703 if c.properties.Sdk_version != "" && ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700704 switch c.properties.Stl {
705 case "":
706 return "ndk_system"
707 case "c++_shared", "c++_static",
708 "stlport_shared", "stlport_static",
709 "gnustl_static":
710 return "ndk_lib" + c.properties.Stl
711 default:
712 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.properties.Stl)
713 return ""
714 }
715 }
716
717 switch c.properties.Stl {
718 case "libc++", "libc++_static",
719 "stlport", "stlport_static",
720 "libstdc++":
721 return c.properties.Stl
722 case "none":
723 return ""
724 case "":
725 if c.shared() {
726 return "libc++" // TODO: mingw needs libstdc++
727 } else {
728 return "libc++_static"
729 }
730 default:
731 ctx.ModuleErrorf("stl: %q is not a supported STL", c.properties.Stl)
732 return ""
733 }
734}
735
736func (c *ccLinked) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
737 stl := c.stl(ctx)
738 if ctx.Failed() {
739 return flags
740 }
741
742 switch stl {
743 case "libc++", "libc++_static":
744 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
745 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/external/libcxx/include")
746 if ctx.Host() {
747 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
748 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
749 flags.LdLibs = append(flags.LdLibs, "-lc", "-lm", "-lpthread")
750 }
751 case "stlport", "stlport_static":
752 if ctx.Device() {
753 flags.IncludeDirs = append(flags.IncludeDirs,
754 "${SrcDir}/external/stlport/stlport",
755 "${SrcDir}/bionic/libstdc++/include",
756 "${SrcDir}/bionic")
757 }
758 case "libstdc++":
759 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
760 // tree is in good enough shape to not need it.
761 // Host builds will use GNU libstdc++.
762 if ctx.Device() {
763 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/bionic/libstdc++/include")
764 }
765 case "ndk_system":
Colin Cross1332b002015-04-07 17:11:30 -0700766 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources/"
Colin Crossed4cf0b2015-03-26 14:43:45 -0700767 flags.IncludeDirs = append(flags.IncludeDirs, ndkSrcRoot+"cxx-stl/system/include")
768 case "ndk_libc++_shared", "ndk_libc++_static":
769 // TODO(danalbert): This really shouldn't be here...
770 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
771 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
772 // Nothing
773 case "":
774 // None or error.
775 if ctx.Host() {
776 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
777 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
778 flags.LdLibs = append(flags.LdLibs, "-lc", "-lm")
779 }
780 default:
781 panic(fmt.Errorf("Unknown stl in ccLinked.Flags: %q", stl))
782 }
783
784 return flags
785}
786
787func (c *ccLinked) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700788 depNames = c.ccBase.DepNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800789
Colin Crossed4cf0b2015-03-26 14:43:45 -0700790 stl := c.stl(ctx)
791 if ctx.Failed() {
792 return depNames
793 }
794
795 switch stl {
796 case "libc++":
797 depNames.SharedLibs = append(depNames.SharedLibs, stl)
798 case "libstdc++":
799 if ctx.Device() {
800 depNames.SharedLibs = append(depNames.SharedLibs, stl)
801 }
802 case "libc++_static":
803 depNames.StaticLibs = append(depNames.StaticLibs, stl)
804 case "stlport":
805 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
806 case "stlport_static":
807 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
808 case "":
809 // None or error.
810 case "ndk_system":
811 // TODO: Make a system STL prebuilt for the NDK.
812 // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have
813 // its own includes. The includes are handled in ccBase.Flags().
Colin Cross577f6e42015-03-27 18:23:34 -0700814 depNames.SharedLibs = append([]string{"libstdc++"}, depNames.SharedLibs...)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700815 case "ndk_libc++_shared", "ndk_libstlport_shared":
816 depNames.SharedLibs = append(depNames.SharedLibs, stl)
817 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
818 depNames.StaticLibs = append(depNames.StaticLibs, stl)
819 default:
820 panic(fmt.Errorf("Unknown stl in ccLinked.DepNames: %q", stl))
821 }
822
Colin Crossf6566ed2015-03-24 11:13:38 -0700823 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700824 if ctx.ModuleName() != "libcompiler_rt-extras" {
825 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
826 }
Colin Cross77b00fa2015-03-16 16:15:49 -0700827 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700828 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700829
830 if c.shared() {
831 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
832 }
Colin Cross577f6e42015-03-27 18:23:34 -0700833
834 if c.properties.Sdk_version != "" {
835 version := c.properties.Sdk_version
836 depNames.SharedLibs = append(depNames.SharedLibs,
837 "ndk_libc."+version,
838 "ndk_libm."+version,
839 )
840 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800841 }
842
Colin Cross21b9a242015-03-24 14:15:58 -0700843 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800844}
845
Colin Crossed4cf0b2015-03-26 14:43:45 -0700846// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
847type ccLinkedInterface interface {
848 // Returns true if the build options for the module have selected a static or shared build
849 buildStatic() bool
850 buildShared() bool
851
852 // Sets whether a specific variant is static or shared
853 setStatic()
854 setShared()
855
856 // Returns whether a specific variant is static or shared
857 static() bool
858 shared() bool
859}
860
861var _ ccLinkedInterface = (*CCLibrary)(nil)
862var _ ccLinkedInterface = (*CCBinary)(nil)
863
864func (c *ccLinked) static() bool {
865 return c.dynamicProperties.VariantIsStatic
866}
867
868func (c *ccLinked) shared() bool {
869 return c.dynamicProperties.VariantIsShared
870}
871
872func (c *ccLinked) setStatic() {
873 c.dynamicProperties.VariantIsStatic = true
874}
875
876func (c *ccLinked) setShared() {
877 c.dynamicProperties.VariantIsShared = true
878}
879
Colin Cross3f40fa42015-01-30 17:27:36 -0800880type ccExportedIncludeDirsProducer interface {
881 exportedIncludeDirs() []string
882}
883
884//
885// Combined static+shared libraries
886//
887
Colin Cross97ba0732015-03-23 17:50:24 -0700888type CCLibrary struct {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700889 ccLinked
Colin Cross3f40fa42015-01-30 17:27:36 -0800890
Colin Crossed4cf0b2015-03-26 14:43:45 -0700891 reuseFrom ccLibraryInterface
892 reuseObjFiles []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800893 objFiles []string
894 exportIncludeDirs []string
895 out string
896
Colin Cross97ba0732015-03-23 17:50:24 -0700897 LibraryProperties struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800898 BuildStatic bool `blueprint:"mutated"`
899 BuildShared bool `blueprint:"mutated"`
Colin Crossed4cf0b2015-03-26 14:43:45 -0700900 Static struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800901 Srcs []string `android:"arch_variant"`
902 Cflags []string `android:"arch_variant"`
903 } `android:"arch_variant"`
904 Shared struct {
905 Srcs []string `android:"arch_variant"`
906 Cflags []string `android:"arch_variant"`
907 } `android:"arch_variant"`
908 }
909}
910
Colin Crossed4cf0b2015-03-26 14:43:45 -0700911func (c *CCLibrary) buildStatic() bool {
912 return c.LibraryProperties.BuildStatic
913}
914
915func (c *CCLibrary) buildShared() bool {
916 return c.LibraryProperties.BuildShared
917}
918
Colin Cross97ba0732015-03-23 17:50:24 -0700919type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700920 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -0700921 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700922 setReuseFrom(ccLibraryInterface)
923 getReuseFrom() ccLibraryInterface
924 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -0700925 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -0700926}
927
Colin Crossed4cf0b2015-03-26 14:43:45 -0700928var _ ccLibraryInterface = (*CCLibrary)(nil)
929
Colin Cross97ba0732015-03-23 17:50:24 -0700930func (c *CCLibrary) ccLibrary() *CCLibrary {
931 return c
Colin Cross3f40fa42015-01-30 17:27:36 -0800932}
933
Colin Cross97ba0732015-03-23 17:50:24 -0700934func NewCCLibrary(library *CCLibrary, module CCModuleType,
935 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
936
Colin Crossed4cf0b2015-03-26 14:43:45 -0700937 return newCCDynamic(&library.ccLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -0700938 &library.LibraryProperties)
939}
940
941func CCLibraryFactory() (blueprint.Module, []interface{}) {
942 module := &CCLibrary{}
943
944 module.LibraryProperties.BuildShared = true
945 module.LibraryProperties.BuildStatic = true
946
947 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
948}
949
Colin Cross21b9a242015-03-24 14:15:58 -0700950func (c *CCLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700951 depNames = c.ccLinked.DepNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -0700952 if c.shared() {
Colin Crossf6566ed2015-03-24 11:13:38 -0700953 if ctx.Device() {
Colin Cross21b9a242015-03-24 14:15:58 -0700954 depNames.CrtBegin = "crtbegin_so"
955 depNames.CrtEnd = "crtend_so"
Colin Cross3f40fa42015-01-30 17:27:36 -0800956 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800957 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800958
Colin Cross21b9a242015-03-24 14:15:58 -0700959 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800960}
961
Colin Cross97ba0732015-03-23 17:50:24 -0700962func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800963 return c.out
964}
965
Colin Crossed4cf0b2015-03-26 14:43:45 -0700966func (c *CCLibrary) getReuseObjFiles() []string {
967 return c.reuseObjFiles
968}
969
970func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
971 c.reuseFrom = reuseFrom
972}
973
974func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
975 return c.reuseFrom
976}
977
Colin Cross97ba0732015-03-23 17:50:24 -0700978func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800979 return c.objFiles
980}
981
Colin Cross97ba0732015-03-23 17:50:24 -0700982func (c *CCLibrary) exportedIncludeDirs() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800983 return c.exportIncludeDirs
984}
985
Colin Cross21b9a242015-03-24 14:15:58 -0700986func (c *CCLibrary) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700987 flags = c.ccLinked.Flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -0700988
Colin Cross97ba0732015-03-23 17:50:24 -0700989 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -0800990
Colin Crossed4cf0b2015-03-26 14:43:45 -0700991 if c.shared() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800992 libName := ctx.ModuleName()
993 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
994 sharedFlag := "-Wl,-shared"
Colin Crossf6566ed2015-03-24 11:13:38 -0700995 if c.properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800996 sharedFlag = "-shared"
997 }
Colin Crossf6566ed2015-03-24 11:13:38 -0700998 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700999 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001000 }
Colin Cross97ba0732015-03-23 17:50:24 -07001001
1002 flags.LdFlags = append(flags.LdFlags,
1003 "-Wl,--gc-sections",
1004 sharedFlag,
1005 "-Wl,-soname,"+libName+sharedLibraryExtension,
1006 )
Colin Cross3f40fa42015-01-30 17:27:36 -08001007 }
Colin Cross97ba0732015-03-23 17:50:24 -07001008
1009 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001010}
1011
Colin Cross97ba0732015-03-23 17:50:24 -07001012func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1013 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001014
1015 staticFlags := flags
Colin Cross97ba0732015-03-23 17:50:24 -07001016 staticFlags.CFlags = append(staticFlags.CFlags, c.LibraryProperties.Static.Cflags...)
Colin Cross581c1892015-04-07 16:50:10 -07001017 objFilesStatic := c.customCompileObjs(ctx, staticFlags, common.DeviceStaticLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001018 c.LibraryProperties.Static.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001019
1020 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001021 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001022
1023 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1024
1025 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1026
1027 c.objFiles = objFiles
1028 c.out = outputFile
1029 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1030 common.ModuleSrcDir(ctx))
1031
1032 ctx.CheckbuildFile(outputFile)
1033}
1034
Colin Cross97ba0732015-03-23 17:50:24 -07001035func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1036 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001037
1038 sharedFlags := flags
Colin Cross97ba0732015-03-23 17:50:24 -07001039 sharedFlags.CFlags = append(sharedFlags.CFlags, c.LibraryProperties.Shared.Cflags...)
Colin Cross581c1892015-04-07 16:50:10 -07001040 objFilesShared := c.customCompileObjs(ctx, sharedFlags, common.DeviceSharedLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001041 c.LibraryProperties.Shared.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001042
1043 objFiles = append(objFiles, objFilesShared...)
1044
1045 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1046
Colin Cross97ba0732015-03-23 17:50:24 -07001047 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001048 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, false,
Colin Cross77b00fa2015-03-16 16:15:49 -07001049 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001050
1051 c.out = outputFile
1052 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1053 common.ModuleSrcDir(ctx))
Colin Cross3f40fa42015-01-30 17:27:36 -08001054}
1055
Colin Cross97ba0732015-03-23 17:50:24 -07001056func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1057 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001058
1059 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001060 if c.getReuseFrom().ccLibrary() == c {
1061 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001062 } else {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001063 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001064 }
1065
Colin Crossed4cf0b2015-03-26 14:43:45 -07001066 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001067 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1068 } else {
1069 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1070 }
1071}
1072
Colin Cross97ba0732015-03-23 17:50:24 -07001073func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001074 // Static libraries do not get installed.
1075}
1076
Colin Cross97ba0732015-03-23 17:50:24 -07001077func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001078 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001079 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001080 installDir = "lib64"
1081 }
1082
1083 ctx.InstallFile(installDir, c.out)
1084}
1085
Colin Cross97ba0732015-03-23 17:50:24 -07001086func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001087 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001088 c.installStaticLibrary(ctx, flags)
1089 } else {
1090 c.installSharedLibrary(ctx, flags)
1091 }
1092}
1093
Colin Cross3f40fa42015-01-30 17:27:36 -08001094//
1095// Objects (for crt*.o)
1096//
1097
1098type ccObject struct {
1099 ccBase
1100 out string
1101}
1102
Colin Cross97ba0732015-03-23 17:50:24 -07001103func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001104 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001105
Colin Crossc472d572015-03-17 15:06:21 -07001106 return newCCBase(&module.ccBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001107}
1108
1109func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1110 // object files can't have any dynamic dependencies
1111 return nil
1112}
1113
Colin Cross21b9a242015-03-24 14:15:58 -07001114func (*ccObject) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1115 // object files can't have any dynamic dependencies
1116 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001117}
1118
1119func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001120 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001121
Colin Cross97ba0732015-03-23 17:50:24 -07001122 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001123
1124 var outputFile string
1125 if len(objFiles) == 1 {
1126 outputFile = objFiles[0]
1127 } else {
1128 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+".o")
1129 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1130 }
1131
1132 c.out = outputFile
1133
1134 ctx.CheckbuildFile(outputFile)
1135}
1136
Colin Cross97ba0732015-03-23 17:50:24 -07001137func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001138 // Object files do not get installed.
1139}
1140
Colin Cross3f40fa42015-01-30 17:27:36 -08001141func (c *ccObject) outputFile() string {
1142 return c.out
1143}
1144
1145//
1146// Executables
1147//
1148
Colin Cross97ba0732015-03-23 17:50:24 -07001149type CCBinary struct {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001150 ccLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001151 out string
Colin Cross97ba0732015-03-23 17:50:24 -07001152 BinaryProperties struct {
1153 // static_executable: compile executable with -static
1154 Static_executable bool
1155
1156 // stem: set the name of the output
1157 Stem string `android:"arch_variant"`
1158
Colin Cross4ae185c2015-03-26 15:12:10 -07001159 // suffix: append to the name of the output
1160 Suffix string `android:"arch_variant"`
1161
Colin Cross97ba0732015-03-23 17:50:24 -07001162 // prefix_symbols: if set, add an extra objcopy --prefix-symbols= step
1163 Prefix_symbols string
1164 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001165}
1166
Colin Crossed4cf0b2015-03-26 14:43:45 -07001167func (c *CCBinary) buildStatic() bool {
1168 return c.BinaryProperties.Static_executable
1169}
1170
1171func (c *CCBinary) buildShared() bool {
1172 return !c.BinaryProperties.Static_executable
1173}
1174
Colin Cross97ba0732015-03-23 17:50:24 -07001175func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001176 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001177 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001178 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001179 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001180
1181 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001182}
1183
Colin Cross21b9a242015-03-24 14:15:58 -07001184func (c *CCBinary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001185 depNames = c.ccLinked.DepNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001186 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001187 if c.BinaryProperties.Static_executable {
Colin Cross21b9a242015-03-24 14:15:58 -07001188 depNames.CrtBegin = "crtbegin_static"
Colin Cross3f40fa42015-01-30 17:27:36 -08001189 } else {
Colin Cross21b9a242015-03-24 14:15:58 -07001190 depNames.CrtBegin = "crtbegin_dynamic"
Colin Cross3f40fa42015-01-30 17:27:36 -08001191 }
Colin Cross21b9a242015-03-24 14:15:58 -07001192 depNames.CrtEnd = "crtend_android"
Colin Crossed4cf0b2015-03-26 14:43:45 -07001193
1194 if c.BinaryProperties.Static_executable {
1195 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1196 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1197 // move them to the beginning of deps.LateStaticLibs
1198 var groupLibs []string
1199 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1200 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1201 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1202 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001203 }
Colin Cross21b9a242015-03-24 14:15:58 -07001204 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001205}
1206
Colin Cross97ba0732015-03-23 17:50:24 -07001207func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001208 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001209
Colin Cross1f8f2342015-03-26 16:09:47 -07001210 props = append(props, &binary.BinaryProperties)
1211
1212 return newCCDynamic(&binary.ccLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001213}
1214
Colin Cross97ba0732015-03-23 17:50:24 -07001215func CCBinaryFactory() (blueprint.Module, []interface{}) {
1216 module := &CCBinary{}
1217
1218 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001219}
1220
Colin Cross21b9a242015-03-24 14:15:58 -07001221func (c *CCBinary) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001222 flags = c.ccLinked.Flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001223
Colin Cross97ba0732015-03-23 17:50:24 -07001224 flags.CFlags = append(flags.CFlags, "-fpie")
1225
Colin Crossf6566ed2015-03-24 11:13:38 -07001226 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001227 if c.BinaryProperties.Static_executable {
1228 // Clang driver needs -static to create static executable.
1229 // However, bionic/linker uses -shared to overwrite.
1230 // Linker for x86 targets does not allow coexistance of -static and -shared,
1231 // so we add -static only if -shared is not used.
1232 if !inList("-shared", flags.LdFlags) {
1233 flags.LdFlags = append(flags.LdFlags, "-static")
1234 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001235
Colin Crossed4cf0b2015-03-26 14:43:45 -07001236 flags.LdFlags = append(flags.LdFlags,
1237 "-nostdlib",
1238 "-Bstatic",
1239 "-Wl,--gc-sections",
1240 )
1241
1242 } else {
1243 linker := "/system/bin/linker"
1244 if flags.Toolchain.Is64Bit() {
1245 linker = "/system/bin/linker64"
1246 }
1247
1248 flags.LdFlags = append(flags.LdFlags,
1249 "-nostdlib",
1250 "-Bdynamic",
1251 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1252 "-Wl,--gc-sections",
1253 "-Wl,-z,nocopyreloc",
1254 )
1255 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001256 }
1257
Colin Cross97ba0732015-03-23 17:50:24 -07001258 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001259}
1260
Colin Cross97ba0732015-03-23 17:50:24 -07001261func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1262 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001263
Colin Cross97ba0732015-03-23 17:50:24 -07001264 if !c.BinaryProperties.Static_executable && inList("libc", c.properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001265 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1266 "from static libs or set static_executable: true")
1267 }
1268
1269 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001270 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001271 if c.BinaryProperties.Prefix_symbols != "" {
1272 afterPrefixSymbols := outputFile
1273 outputFile = outputFile + ".intermediate"
1274 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1275 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1276 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001277
Colin Cross97ba0732015-03-23 17:50:24 -07001278 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001279 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001280 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001281}
Colin Cross3f40fa42015-01-30 17:27:36 -08001282
Colin Cross97ba0732015-03-23 17:50:24 -07001283func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001284 ctx.InstallFile("bin", c.out)
1285}
1286
1287type ccTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001288 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001289
1290 testProperties struct {
1291 // test_per_src: Create a separate test for each source file. Useful when there is
1292 // global state that can not be torn down and reset between each test suite.
1293 Test_per_src bool
1294 }
Dan Albertc403f7c2015-03-18 14:01:18 -07001295}
1296
Colin Cross21b9a242015-03-24 14:15:58 -07001297func (c *ccTest) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
1298 flags = c.CCBinary.Flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001299
Colin Cross97ba0732015-03-23 17:50:24 -07001300 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001301 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001302 flags.CFlags = append(flags.CFlags, "-O0", "-g")
1303 flags.LdLibs = append(flags.LdLibs, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001304 }
1305
1306 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross97ba0732015-03-23 17:50:24 -07001307 flags.IncludeDirs = append(flags.IncludeDirs,
Colin Cross1332b002015-04-07 17:11:30 -07001308 filepath.Join(ctx.AConfig().SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001309
Colin Cross21b9a242015-03-24 14:15:58 -07001310 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001311}
1312
Colin Cross21b9a242015-03-24 14:15:58 -07001313func (c *ccTest) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1314 depNames = c.CCBinary.DepNames(ctx, depNames)
1315 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
1316 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001317}
1318
Colin Cross97ba0732015-03-23 17:50:24 -07001319func (c *ccTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001320 if ctx.Device() {
Tim Kilbourn5ccc7302015-03-19 10:02:21 -07001321 ctx.InstallFile("../data/nativetest/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001322 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001323 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001324 }
1325}
1326
Colin Cross97ba0732015-03-23 17:50:24 -07001327func CCTestFactory() (blueprint.Module, []interface{}) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001328 module := &ccTest{}
Colin Cross1f8f2342015-03-26 16:09:47 -07001329 return NewCCBinary(&module.CCBinary, module, common.HostAndDeviceSupported,
1330 &module.testProperties)
Colin Cross6b290692015-03-19 14:05:33 -07001331}
1332
1333func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
1334 if test, ok := mctx.Module().(*ccTest); ok {
1335 if test.testProperties.Test_per_src {
1336 testNames := make([]string, len(test.properties.Srcs))
1337 for i, src := range test.properties.Srcs {
1338 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1339 }
1340 tests := mctx.CreateLocalVariations(testNames...)
1341 for i, src := range test.properties.Srcs {
1342 tests[i].(*ccTest).properties.Srcs = []string{src}
Colin Cross97ba0732015-03-23 17:50:24 -07001343 tests[i].(*ccTest).BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001344 }
1345 }
1346 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001347}
1348
1349//
1350// Static library
1351//
1352
Colin Cross97ba0732015-03-23 17:50:24 -07001353func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1354 module := &CCLibrary{}
1355 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001356
Colin Cross97ba0732015-03-23 17:50:24 -07001357 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001358}
1359
1360//
1361// Shared libraries
1362//
1363
Colin Cross97ba0732015-03-23 17:50:24 -07001364func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1365 module := &CCLibrary{}
1366 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001367
Colin Cross97ba0732015-03-23 17:50:24 -07001368 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001369}
1370
1371//
1372// Host static library
1373//
1374
Colin Cross97ba0732015-03-23 17:50:24 -07001375func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1376 module := &CCLibrary{}
1377 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001378
Colin Cross97ba0732015-03-23 17:50:24 -07001379 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001380}
1381
1382//
1383// Host Shared libraries
1384//
1385
Colin Cross97ba0732015-03-23 17:50:24 -07001386func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1387 module := &CCLibrary{}
1388 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001389
Colin Cross97ba0732015-03-23 17:50:24 -07001390 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001391}
1392
1393//
1394// Host Binaries
1395//
1396
Colin Cross97ba0732015-03-23 17:50:24 -07001397func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1398 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001399
Colin Cross97ba0732015-03-23 17:50:24 -07001400 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001401}
1402
1403//
Colin Cross1f8f2342015-03-26 16:09:47 -07001404// Host Tests
1405//
1406
1407func CCTestHostFactory() (blueprint.Module, []interface{}) {
1408 module := &ccTest{}
1409 return NewCCBinary(&module.CCBinary, module, common.HostSupported,
1410 &module.testProperties)
1411}
1412
1413//
Colin Cross3f40fa42015-01-30 17:27:36 -08001414// Device libraries shipped with gcc
1415//
1416
1417type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001418 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001419}
1420
1421func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1422 // toolchain libraries can't have any dependencies
1423 return nil
1424}
1425
Colin Cross21b9a242015-03-24 14:15:58 -07001426func (*toolchainLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001427 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001428 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001429}
1430
Colin Cross97ba0732015-03-23 17:50:24 -07001431func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001432 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001433
Colin Cross97ba0732015-03-23 17:50:24 -07001434 module.LibraryProperties.BuildStatic = true
1435
Colin Cross21b9a242015-03-24 14:15:58 -07001436 return newCCBase(&module.ccBase, module, common.DeviceSupported, common.MultilibBoth,
1437 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001438}
1439
1440func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001441 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001442
1443 libName := ctx.ModuleName() + staticLibraryExtension
1444 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1445
1446 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1447
1448 c.out = outputFile
1449
1450 ctx.CheckbuildFile(outputFile)
1451}
1452
Colin Cross97ba0732015-03-23 17:50:24 -07001453func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001454 // Toolchain libraries do not get installed.
1455}
1456
Dan Albertbe961682015-03-18 23:38:50 -07001457// NDK prebuilt libraries.
1458//
1459// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1460// either (with the exception of the shared STLs, which are installed to the app's directory rather
1461// than to the system image).
1462
1463func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1464 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
Colin Cross1332b002015-04-07 17:11:30 -07001465 ctx.AConfig().SrcDir(), version, toolchain.Name())
Dan Albertbe961682015-03-18 23:38:50 -07001466}
1467
1468type ndkPrebuiltLibrary struct {
1469 CCLibrary
1470}
1471
1472func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1473 ctx common.AndroidDynamicDependerModuleContext) []string {
1474
1475 // NDK libraries can't have any dependencies
1476 return nil
1477}
1478
1479func (*ndkPrebuiltLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1480 // NDK libraries can't have any dependencies
1481 return CCDeps{}
1482}
1483
1484func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1485 module := &ndkPrebuiltLibrary{}
1486 module.LibraryProperties.BuildShared = true
1487 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1488}
1489
1490func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1491 deps CCDeps, objFiles []string) {
1492 // A null build step, but it sets up the output path.
1493 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1494 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1495 }
1496
1497 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1498 common.ModuleSrcDir(ctx))
1499
1500 // NDK prebuilt libraries are named like: ndk_LIBNAME.SDK_VERSION.
1501 // We want to translate to just LIBNAME.
1502 libName := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1503 libDir := getNdkLibDir(ctx, flags.Toolchain, c.properties.Sdk_version)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001504 c.out = filepath.Join(libDir, libName+sharedLibraryExtension)
Dan Albertbe961682015-03-18 23:38:50 -07001505}
1506
1507func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1508 // Toolchain libraries do not get installed.
1509}
1510
1511// The NDK STLs are slightly different from the prebuilt system libraries:
1512// * Are not specific to each platform version.
1513// * The libraries are not in a predictable location for each STL.
1514
1515type ndkPrebuiltStl struct {
1516 ndkPrebuiltLibrary
1517}
1518
1519type ndkPrebuiltStaticStl struct {
1520 ndkPrebuiltStl
1521}
1522
1523type ndkPrebuiltSharedStl struct {
1524 ndkPrebuiltStl
1525}
1526
1527func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1528 module := &ndkPrebuiltSharedStl{}
1529 module.LibraryProperties.BuildShared = true
1530 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1531}
1532
1533func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1534 module := &ndkPrebuiltStaticStl{}
1535 module.LibraryProperties.BuildStatic = true
1536 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1537}
1538
1539func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1540 gccVersion := toolchain.GccVersion()
1541 var libDir string
1542 switch stl {
1543 case "libstlport":
1544 libDir = "cxx-stl/stlport/libs"
1545 case "libc++":
1546 libDir = "cxx-stl/llvm-libc++/libs"
1547 case "libgnustl":
1548 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1549 }
1550
1551 if libDir != "" {
Colin Cross1332b002015-04-07 17:11:30 -07001552 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources"
Dan Albertbe961682015-03-18 23:38:50 -07001553 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1554 }
1555
1556 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1557 return ""
1558}
1559
1560func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1561 deps CCDeps, objFiles []string) {
1562 // A null build step, but it sets up the output path.
1563 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1564 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1565 }
1566
1567 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1568 common.ModuleSrcDir(ctx))
1569
1570 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1571 libExt := sharedLibraryExtension
1572 if c.LibraryProperties.BuildStatic {
1573 libExt = staticLibraryExtension
1574 }
1575
1576 stlName := strings.TrimSuffix(libName, "_shared")
1577 stlName = strings.TrimSuffix(stlName, "_static")
1578 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1579 c.out = libDir + "/" + libName + libExt
1580}
1581
Colin Cross3f40fa42015-01-30 17:27:36 -08001582func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001583 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001584 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001585 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001586 modules = mctx.CreateLocalVariations("static", "shared")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001587 modules[0].(ccLinkedInterface).setStatic()
1588 modules[1].(ccLinkedInterface).setShared()
1589 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001590 modules = mctx.CreateLocalVariations("static")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001591 modules[0].(ccLinkedInterface).setStatic()
1592 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001593 modules = mctx.CreateLocalVariations("shared")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001594 modules[0].(ccLinkedInterface).setShared()
Colin Cross3f40fa42015-01-30 17:27:36 -08001595 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001596 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001597 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001598
1599 if _, ok := c.(ccLibraryInterface); ok {
1600 reuseFrom := modules[0].(ccLibraryInterface)
1601 for _, m := range modules {
1602 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001603 }
1604 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001605 }
1606}