blob: b0dd7920a6c13f8500632f0c48547450b5a04bb4 [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
33type Config interface {
34 SrcDir() string
Colin Cross581c1892015-04-07 16:50:10 -070035 IntermediatesDir() string
Colin Cross3f40fa42015-01-30 17:27:36 -080036 PrebuiltOS() string
37}
38
39var (
40 HostPrebuiltTag = pctx.VariableConfigMethod("HostPrebuiltTag", Config.PrebuiltOS)
41 SrcDir = pctx.VariableConfigMethod("SrcDir", Config.SrcDir)
42
43 LibcRoot = pctx.StaticVariable("LibcRoot", "${SrcDir}/bionic/libc")
44 LibmRoot = pctx.StaticVariable("LibmRoot", "${SrcDir}/bionic/libm")
45)
46
47// Flags used by lots of devices. Putting them in package static variables will save bytes in
48// build.ninja so they aren't repeated for every file
49var (
50 commonGlobalCflags = []string{
51 "-DANDROID",
52 "-fmessage-length=0",
53 "-W",
54 "-Wall",
55 "-Wno-unused",
56 "-Winit-self",
57 "-Wpointer-arith",
58
59 // COMMON_RELEASE_CFLAGS
60 "-DNDEBUG",
61 "-UDEBUG",
62 }
63
64 deviceGlobalCflags = []string{
65 // TARGET_ERROR_FLAGS
66 "-Werror=return-type",
67 "-Werror=non-virtual-dtor",
68 "-Werror=address",
69 "-Werror=sequence-point",
70 }
71
72 hostGlobalCflags = []string{}
73
74 commonGlobalCppflags = []string{
75 "-Wsign-promo",
76 "-std=gnu++11",
77 }
78)
79
80func init() {
81 pctx.StaticVariable("commonGlobalCflags", strings.Join(commonGlobalCflags, " "))
82 pctx.StaticVariable("deviceGlobalCflags", strings.Join(deviceGlobalCflags, " "))
83 pctx.StaticVariable("hostGlobalCflags", strings.Join(hostGlobalCflags, " "))
84
85 pctx.StaticVariable("commonGlobalCppflags", strings.Join(commonGlobalCppflags, " "))
86
87 pctx.StaticVariable("commonClangGlobalCflags",
88 strings.Join(clangFilterUnknownCflags(commonGlobalCflags), " "))
89 pctx.StaticVariable("deviceClangGlobalCflags",
90 strings.Join(clangFilterUnknownCflags(deviceGlobalCflags), " "))
91 pctx.StaticVariable("hostClangGlobalCflags",
92 strings.Join(clangFilterUnknownCflags(hostGlobalCflags), " "))
Tim Kilbournf2948142015-03-11 12:03:03 -070093 pctx.StaticVariable("commonClangGlobalCppflags",
94 strings.Join(clangFilterUnknownCflags(commonGlobalCppflags), " "))
Colin Cross3f40fa42015-01-30 17:27:36 -080095
96 // Everything in this list is a crime against abstraction and dependency tracking.
97 // Do not add anything to this list.
98 pctx.StaticVariable("commonGlobalIncludes", strings.Join([]string{
99 "-isystem ${SrcDir}/system/core/include",
100 "-isystem ${SrcDir}/hardware/libhardware/include",
101 "-isystem ${SrcDir}/hardware/libhardware_legacy/include",
102 "-isystem ${SrcDir}/hardware/ril/include",
103 "-isystem ${SrcDir}/libnativehelper/include",
104 "-isystem ${SrcDir}/frameworks/native/include",
105 "-isystem ${SrcDir}/frameworks/native/opengl/include",
106 "-isystem ${SrcDir}/frameworks/av/include",
107 "-isystem ${SrcDir}/frameworks/base/include",
108 }, " "))
109
110 pctx.StaticVariable("clangPath", "${SrcDir}/prebuilts/clang/${HostPrebuiltTag}/host/3.6/bin/")
111}
112
Colin Cross97ba0732015-03-23 17:50:24 -0700113// ccProperties describes properties used to compile all C or C++ modules
Colin Cross3f40fa42015-01-30 17:27:36 -0800114type ccProperties struct {
115 // srcs: list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
116 Srcs []string `android:"arch_variant,arch_subtract"`
117
118 // cflags: list of module-specific flags that will be used for C and C++ compiles.
119 Cflags []string `android:"arch_variant"`
120
121 // cppflags: list of module-specific flags that will be used for C++ compiles
122 Cppflags []string `android:"arch_variant"`
123
124 // conlyflags: list of module-specific flags that will be used for C compiles
125 Conlyflags []string `android:"arch_variant"`
126
127 // asflags: list of module-specific flags that will be used for .S compiles
128 Asflags []string `android:"arch_variant"`
129
Colin Cross581c1892015-04-07 16:50:10 -0700130 // yaccflags: list of module-specific flags that will be used for .y and .yy compiles
131 Yaccflags []string
132
Colin Cross3f40fa42015-01-30 17:27:36 -0800133 // ldflags: list of module-specific flags that will be used for all link steps
134 Ldflags []string `android:"arch_variant"`
135
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700136 // instruction_set: the instruction set architecture to use to compile the C/C++
137 // module.
138 Instruction_set string `android:"arch_variant"`
139
Colin Cross3f40fa42015-01-30 17:27:36 -0800140 // include_dirs: list of directories relative to the root of the source tree that will
141 // be added to the include path using -I.
142 // If possible, don't use this. If adding paths from the current directory use
143 // local_include_dirs, if adding paths from other modules use export_include_dirs in
144 // that module.
145 Include_dirs []string `android:"arch_variant"`
146
147 // local_include_dirs: list of directories relative to the Blueprints file that will
148 // be added to the include path using -I
149 Local_include_dirs []string `android:"arch_variant"`
150
151 // export_include_dirs: list of directories relative to the Blueprints file that will
152 // be added to the include path using -I for any module that links against this module
Dan Albertbe961682015-03-18 23:38:50 -0700153 Export_include_dirs []string `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800154
155 // clang_cflags: list of module-specific flags that will be used for C and C++ compiles when
156 // compiling with clang
157 Clang_cflags []string `android:"arch_variant"`
158
159 // clang_asflags: list of module-specific flags that will be used for .S compiles when
160 // compiling with clang
161 Clang_asflags []string `android:"arch_variant"`
162
163 // system_shared_libs: list of system libraries that will be dynamically linked to
164 // shared library and executable modules. If unset, generally defaults to libc
165 // and libm. Set to [] to prevent linking against libc and libm.
166 System_shared_libs []string
167
168 // whole_static_libs: list of modules whose object files should be linked into this module
169 // in their entirety. For static library modules, all of the .o files from the intermediate
170 // directory of the dependency will be linked into this modules .a file. For a shared library,
171 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
172 Whole_static_libs []string `android:"arch_variant"`
173
174 // static_libs: list of modules that should be statically linked into this module.
175 Static_libs []string `android:"arch_variant"`
176
177 // shared_libs: list of modules that should be dynamically linked into this module.
178 Shared_libs []string `android:"arch_variant"`
179
180 // allow_undefined_symbols: allow the module to contain undefined symbols. By default,
181 // modules cannot contain undefined symbols that are not satisified by their immediate
182 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
183 // This flag should only be necessary for compiling low-level libraries like libc.
184 Allow_undefined_symbols bool
185
186 // nocrt: don't link in crt_begin and crt_end. This flag should only be necessary for
187 // compiling crt or libc.
188 Nocrt bool `android:"arch_variant"`
189
190 // no_default_compiler_flags: don't insert default compiler flags into asflags, cflags,
191 // cppflags, conlyflags, ldflags, or include_dirs
192 No_default_compiler_flags bool
193
194 // clang: compile module with clang instead of gcc
195 Clang bool `android:"arch_variant"`
196
197 // rtti: pass -frtti instead of -fno-rtti
198 Rtti bool
199
200 // host_ldlibs: -l arguments to pass to linker for host-provided shared libraries
201 Host_ldlibs []string `android:"arch_variant"`
202
203 // stl: select the STL library to use. Possible values are "libc++", "libc++_static",
204 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
205 // default
206 Stl string
207
208 // Set for combined shared/static libraries to prevent compiling object files a second time
209 SkipCompileObjs bool `blueprint:"mutated"`
Colin Crossaf19a292015-03-18 12:07:10 -0700210
211 Debug struct {
212 Cflags []string `android:"arch_variant"`
213 } `android:"arch_variant"`
214 Release struct {
215 Cflags []string `android:"arch_variant"`
216 } `android:"arch_variant"`
Colin Crossefd8e482015-03-18 17:17:35 -0700217
218 // Minimum sdk version supported when compiling against the ndk
219 Sdk_version string
Colin Cross3f40fa42015-01-30 17:27:36 -0800220}
221
222type unusedProperties struct {
223 Asan bool
224 Native_coverage bool
225 Strip string
226 Tags []string
227 Required []string
228}
229
230// Building C/C++ code is handled by objects that satisfy this interface via composition
Colin Cross97ba0732015-03-23 17:50:24 -0700231type CCModuleType interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800232 common.AndroidModule
233
Colin Cross21b9a242015-03-24 14:15:58 -0700234 // Modify the ccFlags
235 Flags(common.AndroidModuleContext, CCFlags) CCFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800236
Colin Cross21b9a242015-03-24 14:15:58 -0700237 // Return list of dependency names for use in AndroidDynamicDependencies and in depsToPaths
238 DepNames(common.AndroidBaseContext, CCDeps) CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800239
240 // Compile objects into final module
Colin Cross97ba0732015-03-23 17:50:24 -0700241 compileModule(common.AndroidModuleContext, CCFlags, CCDeps, []string)
Colin Cross3f40fa42015-01-30 17:27:36 -0800242
Dan Albertc403f7c2015-03-18 14:01:18 -0700243 // Install the built module.
Colin Cross97ba0732015-03-23 17:50:24 -0700244 installModule(common.AndroidModuleContext, CCFlags)
Dan Albertc403f7c2015-03-18 14:01:18 -0700245
Colin Cross3f40fa42015-01-30 17:27:36 -0800246 // Return the output file (.o, .a or .so) for use by other modules
247 outputFile() string
248}
249
Colin Cross97ba0732015-03-23 17:50:24 -0700250type CCDeps struct {
251 StaticLibs, SharedLibs, LateStaticLibs, WholeStaticLibs, ObjFiles, IncludeDirs []string
Colin Crossc472d572015-03-17 15:06:21 -0700252
Colin Cross21b9a242015-03-24 14:15:58 -0700253 WholeStaticLibObjFiles []string
254
Colin Cross97ba0732015-03-23 17:50:24 -0700255 CrtBegin, CrtEnd string
Colin Crossc472d572015-03-17 15:06:21 -0700256}
257
Colin Cross97ba0732015-03-23 17:50:24 -0700258type CCFlags struct {
259 GlobalFlags []string
260 AsFlags []string
261 CFlags []string
262 ConlyFlags []string
263 CppFlags []string
Colin Cross581c1892015-04-07 16:50:10 -0700264 YaccFlags []string
Colin Cross97ba0732015-03-23 17:50:24 -0700265 LdFlags []string
266 LdLibs []string
267 IncludeDirs []string
268 Nocrt bool
269 Toolchain Toolchain
270 Clang bool
Colin Crossc472d572015-03-17 15:06:21 -0700271}
272
273// ccBase contains the properties and members used by all C/C++ module types, and implements
274// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
275// and uses a ccModuleType interface to that struct to create the build steps.
276type ccBase struct {
277 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700278 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700279
280 properties ccProperties
281 unused unusedProperties
282
283 installPath string
284}
285
Colin Cross97ba0732015-03-23 17:50:24 -0700286func newCCBase(base *ccBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700287 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
288
289 base.module = module
290
291 props = append(props, &base.properties, &base.unused)
292
Colin Cross5049f022015-03-18 13:28:46 -0700293 return common.InitAndroidArchModule(module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700294}
295
Colin Cross3f40fa42015-01-30 17:27:36 -0800296func (c *ccBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
297 toolchain := c.findToolchain(ctx)
298 if ctx.Failed() {
299 return
300 }
301
Colin Cross21b9a242015-03-24 14:15:58 -0700302 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800303 if ctx.Failed() {
304 return
305 }
306
Colin Cross21b9a242015-03-24 14:15:58 -0700307 depNames := c.module.DepNames(ctx, CCDeps{})
Colin Cross3f40fa42015-01-30 17:27:36 -0800308 if ctx.Failed() {
309 return
310 }
311
Colin Cross21b9a242015-03-24 14:15:58 -0700312 deps := c.depsToPaths(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800313 if ctx.Failed() {
314 return
315 }
316
Colin Cross97ba0732015-03-23 17:50:24 -0700317 flags.IncludeDirs = append(flags.IncludeDirs, deps.IncludeDirs...)
Colin Crossed9f8682015-03-18 17:17:35 -0700318
Colin Cross581c1892015-04-07 16:50:10 -0700319 objFiles := c.compileObjs(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800320 if ctx.Failed() {
321 return
322 }
323
Colin Cross581c1892015-04-07 16:50:10 -0700324 generatedObjFiles := c.compileGeneratedObjs(ctx, flags)
Colin Cross5049f022015-03-18 13:28:46 -0700325 if ctx.Failed() {
326 return
327 }
328
329 objFiles = append(objFiles, generatedObjFiles...)
330
Colin Cross3f40fa42015-01-30 17:27:36 -0800331 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
332 if ctx.Failed() {
333 return
334 }
Dan Albertc403f7c2015-03-18 14:01:18 -0700335
336 c.ccModuleType().installModule(ctx, flags)
337 if ctx.Failed() {
338 return
339 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800340}
341
Colin Cross97ba0732015-03-23 17:50:24 -0700342func (c *ccBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800343 return c.module
344}
345
346var _ common.AndroidDynamicDepender = (*ccBase)(nil)
347
Colin Cross97ba0732015-03-23 17:50:24 -0700348func (c *ccBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800349 arch := ctx.Arch()
350 factory := toolchainFactories[arch.HostOrDevice][arch.ArchType]
351 if factory == nil {
352 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
353 arch.HostOrDevice.String(), arch.String()))
354 }
355 return factory(arch.ArchVariant, arch.CpuVariant)
356}
357
Colin Cross21b9a242015-03-24 14:15:58 -0700358func (c *ccBase) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
359 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.properties.Whole_static_libs...)
360 depNames.StaticLibs = append(depNames.StaticLibs, c.properties.Static_libs...)
361 depNames.SharedLibs = append(depNames.SharedLibs, c.properties.Shared_libs...)
362
Colin Cross21b9a242015-03-24 14:15:58 -0700363 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800364}
365
366func (c *ccBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
Colin Cross21b9a242015-03-24 14:15:58 -0700367 depNames := CCDeps{}
368 depNames = c.module.DepNames(ctx, depNames)
369 staticLibs := depNames.WholeStaticLibs
370 staticLibs = append(staticLibs, depNames.StaticLibs...)
371 staticLibs = append(staticLibs, depNames.LateStaticLibs...)
372 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800373
Colin Cross21b9a242015-03-24 14:15:58 -0700374 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, depNames.SharedLibs...)
375
376 ret := append([]string(nil), depNames.ObjFiles...)
377 if depNames.CrtBegin != "" {
378 ret = append(ret, depNames.CrtBegin)
379 }
380 if depNames.CrtEnd != "" {
381 ret = append(ret, depNames.CrtEnd)
382 }
383
384 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800385}
386
387// Create a ccFlags struct that collects the compile flags from global values,
388// per-target values, module type values, and per-module Blueprints properties
Colin Cross21b9a242015-03-24 14:15:58 -0700389func (c *ccBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700390 flags := CCFlags{
391 CFlags: c.properties.Cflags,
392 CppFlags: c.properties.Cppflags,
393 ConlyFlags: c.properties.Conlyflags,
394 LdFlags: c.properties.Ldflags,
395 AsFlags: c.properties.Asflags,
Colin Cross581c1892015-04-07 16:50:10 -0700396 YaccFlags: c.properties.Yaccflags,
Colin Cross97ba0732015-03-23 17:50:24 -0700397 Nocrt: c.properties.Nocrt,
398 Toolchain: toolchain,
399 Clang: c.properties.Clang,
Colin Cross3f40fa42015-01-30 17:27:36 -0800400 }
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700401 instructionSet := c.properties.Instruction_set
402 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
403 if err != nil {
404 ctx.ModuleErrorf("%s", err)
405 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800406
Colin Crossaf19a292015-03-18 12:07:10 -0700407 // TODO: debug
Colin Cross97ba0732015-03-23 17:50:24 -0700408 flags.CFlags = append(flags.CFlags, c.properties.Release.Cflags...)
Colin Crossaf19a292015-03-18 12:07:10 -0700409
Colin Cross28d76592015-03-26 16:14:04 -0700410 if ctx.Host() && !ctx.ContainsProperty("clang") {
Colin Cross97ba0732015-03-23 17:50:24 -0700411 flags.Clang = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800412 }
413
Colin Cross97ba0732015-03-23 17:50:24 -0700414 if flags.Clang {
415 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
416 flags.CFlags = append(flags.CFlags, c.properties.Clang_cflags...)
417 flags.AsFlags = append(flags.AsFlags, c.properties.Clang_asflags...)
418 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
419 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
420 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800421
Colin Cross97ba0732015-03-23 17:50:24 -0700422 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
423 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700424 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700425 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700426 }
427
Colin Cross3f40fa42015-01-30 17:27:36 -0800428 target := "-target " + toolchain.ClangTriple()
429 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
430
Colin Cross97ba0732015-03-23 17:50:24 -0700431 flags.CFlags = append(flags.CFlags, target, gccPrefix)
432 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
433 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800434
Colin Crossf6566ed2015-03-24 11:13:38 -0700435 if ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800436 gccToolchain := "--gcc-toolchain=" + toolchain.GccRoot()
437 sysroot := "--sysroot=" + filepath.Join(toolchain.GccRoot(), "sysroot")
438
439 // TODO: also need more -B, -L flags to make host builds hermetic
Colin Cross97ba0732015-03-23 17:50:24 -0700440 flags.CFlags = append(flags.CFlags, gccToolchain, sysroot)
441 flags.AsFlags = append(flags.AsFlags, gccToolchain, sysroot)
442 flags.LdFlags = append(flags.LdFlags, gccToolchain, sysroot)
Colin Cross3f40fa42015-01-30 17:27:36 -0800443 }
444 }
445
Colin Cross97ba0732015-03-23 17:50:24 -0700446 flags.IncludeDirs = pathtools.PrefixPaths(c.properties.Include_dirs, ctx.Config().(Config).SrcDir())
Colin Cross3f40fa42015-01-30 17:27:36 -0800447 localIncludeDirs := pathtools.PrefixPaths(c.properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross97ba0732015-03-23 17:50:24 -0700448 flags.IncludeDirs = append(flags.IncludeDirs, localIncludeDirs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800449
450 if !c.properties.No_default_compiler_flags {
Colin Cross97ba0732015-03-23 17:50:24 -0700451 flags.IncludeDirs = append(flags.IncludeDirs, []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800452 common.ModuleSrcDir(ctx),
453 common.ModuleOutDir(ctx),
454 common.ModuleGenDir(ctx),
455 }...)
456
Colin Crossefd8e482015-03-18 17:17:35 -0700457 if c.properties.Sdk_version == "" {
Colin Cross97ba0732015-03-23 17:50:24 -0700458 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/libnativehelper/include/nativehelper")
Colin Crossefd8e482015-03-18 17:17:35 -0700459 }
460
Colin Crossf6566ed2015-03-24 11:13:38 -0700461 if ctx.Device() && !c.properties.Allow_undefined_symbols {
Colin Cross97ba0732015-03-23 17:50:24 -0700462 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800463 }
464
Colin Cross97ba0732015-03-23 17:50:24 -0700465 if flags.Clang {
466 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
467 flags.GlobalFlags = []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800468 "${commonGlobalIncludes}",
469 toolchain.IncludeFlags(),
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700470 instructionSetFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800471 toolchain.ClangCflags(),
472 "${commonClangGlobalCflags}",
Colin Crossf6566ed2015-03-24 11:13:38 -0700473 fmt.Sprintf("${%sClangGlobalCflags}", ctx.Arch().HostOrDevice),
Colin Cross3f40fa42015-01-30 17:27:36 -0800474 }
475 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700476 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
477 flags.GlobalFlags = []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800478 "${commonGlobalIncludes}",
479 toolchain.IncludeFlags(),
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700480 instructionSetFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800481 toolchain.Cflags(),
482 "${commonGlobalCflags}",
Colin Crossf6566ed2015-03-24 11:13:38 -0700483 fmt.Sprintf("${%sGlobalCflags}", ctx.Arch().HostOrDevice),
Colin Cross3f40fa42015-01-30 17:27:36 -0800484 }
485 }
486
Colin Crossf6566ed2015-03-24 11:13:38 -0700487 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -0700488 flags.LdFlags = append(flags.LdFlags, c.properties.Host_ldlibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800489 }
490
Colin Crossf6566ed2015-03-24 11:13:38 -0700491 if ctx.Device() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800492 if c.properties.Rtti {
Colin Cross97ba0732015-03-23 17:50:24 -0700493 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800494 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700495 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800496 }
497 }
498
Colin Cross97ba0732015-03-23 17:50:24 -0700499 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800500
Colin Cross97ba0732015-03-23 17:50:24 -0700501 if flags.Clang {
502 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
503 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800504 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700505 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
506 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800507 }
508 }
509
Colin Cross21b9a242015-03-24 14:15:58 -0700510 flags = c.ccModuleType().Flags(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800511
512 // Optimization to reduce size of build.ninja
513 // Replace the long list of flags for each file with a module-local variable
Colin Cross97ba0732015-03-23 17:50:24 -0700514 ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " "))
515 ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " "))
516 ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " "))
517 flags.CFlags = []string{"$cflags"}
518 flags.CppFlags = []string{"$cppflags"}
519 flags.AsFlags = []string{"$asflags"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800520
521 return flags
522}
523
Colin Cross21b9a242015-03-24 14:15:58 -0700524func (c *ccBase) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross3f40fa42015-01-30 17:27:36 -0800525 return flags
526}
527
528// Compile a list of source files into objects a specified subdirectory
Colin Cross97ba0732015-03-23 17:50:24 -0700529func (c *ccBase) customCompileObjs(ctx common.AndroidModuleContext, flags CCFlags,
Colin Cross581c1892015-04-07 16:50:10 -0700530 subdir string, srcFiles []string) []string {
531
532 buildFlags := ccFlagsToBuilderFlags(flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800533
534 srcFiles = pathtools.PrefixPaths(srcFiles, common.ModuleSrcDir(ctx))
535 srcFiles = common.ExpandGlobs(ctx, srcFiles)
Colin Cross581c1892015-04-07 16:50:10 -0700536 srcFiles, deps := genSources(ctx, srcFiles, buildFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800537
Colin Cross581c1892015-04-07 16:50:10 -0700538 return TransformSourceToObj(ctx, subdir, srcFiles, buildFlags, deps)
Colin Cross3f40fa42015-01-30 17:27:36 -0800539}
540
541// Compile files listed in c.properties.Srcs into objects
Colin Cross581c1892015-04-07 16:50:10 -0700542func (c *ccBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800543
544 if c.properties.SkipCompileObjs {
545 return nil
546 }
547
Colin Cross581c1892015-04-07 16:50:10 -0700548 return c.customCompileObjs(ctx, flags, "", c.properties.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800549}
550
Colin Cross5049f022015-03-18 13:28:46 -0700551// Compile generated source files from dependencies
Colin Cross581c1892015-04-07 16:50:10 -0700552func (c *ccBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700553 var srcs []string
554
555 if c.properties.SkipCompileObjs {
556 return nil
557 }
558
559 ctx.VisitDirectDeps(func(module blueprint.Module) {
560 if gen, ok := module.(genrule.SourceFileGenerator); ok {
561 srcs = append(srcs, gen.GeneratedSourceFiles()...)
562 }
563 })
564
565 if len(srcs) == 0 {
566 return nil
567 }
568
Colin Cross581c1892015-04-07 16:50:10 -0700569 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags), nil)
Colin Cross5049f022015-03-18 13:28:46 -0700570}
571
Colin Cross3f40fa42015-01-30 17:27:36 -0800572func (c *ccBase) outputFile() string {
573 return ""
574}
575
Colin Cross21b9a242015-03-24 14:15:58 -0700576func (c *ccBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800577 names []string) (modules []common.AndroidModule,
578 outputFiles []string, exportedIncludeDirs []string) {
579
580 for _, n := range names {
581 found := false
582 ctx.VisitDirectDeps(func(m blueprint.Module) {
583 otherName := ctx.OtherModuleName(m)
584 if otherName != n {
585 return
586 }
587
Colin Cross97ba0732015-03-23 17:50:24 -0700588 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800589 if a.Disabled() {
590 // If a cc_library host+device module depends on a library that exists as both
591 // cc_library_shared and cc_library_host_shared, it will end up with two
592 // dependencies with the same name, one of which is marked disabled for each
593 // of host and device. Ignore the disabled one.
594 return
595 }
596 if a.HostOrDevice() != ctx.Arch().HostOrDevice {
597 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
598 otherName)
599 return
600 }
601
602 if outputFile := a.outputFile(); outputFile != "" {
603 if found {
604 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
605 return
606 }
607 outputFiles = append(outputFiles, outputFile)
608 modules = append(modules, a)
609 if i, ok := a.(ccExportedIncludeDirsProducer); ok {
610 exportedIncludeDirs = append(exportedIncludeDirs, i.exportedIncludeDirs()...)
611 }
612 found = true
613 } else {
614 ctx.ModuleErrorf("module %q missing output file", otherName)
615 return
616 }
617 } else {
618 ctx.ModuleErrorf("module %q not an android module", otherName)
619 return
620 }
621 })
622 if !found {
623 ctx.ModuleErrorf("unsatisified dependency on %q", n)
624 }
625 }
626
627 return modules, outputFiles, exportedIncludeDirs
628}
629
Colin Cross21b9a242015-03-24 14:15:58 -0700630// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
631// containing paths
632func (c *ccBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
633 var depPaths CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800634 var newIncludeDirs []string
635
Colin Cross21b9a242015-03-24 14:15:58 -0700636 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800637
Colin Cross21b9a242015-03-24 14:15:58 -0700638 wholeStaticLibModules, depPaths.WholeStaticLibs, newIncludeDirs =
639 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
640 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800641
Colin Cross21b9a242015-03-24 14:15:58 -0700642 for _, m := range wholeStaticLibModules {
643 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
644 depPaths.WholeStaticLibObjFiles =
645 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
646 } else {
647 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
648 }
649 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800650
Colin Cross21b9a242015-03-24 14:15:58 -0700651 _, depPaths.StaticLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.StaticLibs)
652 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
653
654 _, depPaths.LateStaticLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
655 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
656
657 _, depPaths.SharedLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.SharedLibs)
658 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
659
660 ctx.VisitDirectDeps(func(m blueprint.Module) {
661 if obj, ok := m.(*ccObject); ok {
662 otherName := ctx.OtherModuleName(m)
663 if otherName == depNames.CrtBegin {
664 if !c.properties.Nocrt {
665 depPaths.CrtBegin = obj.outputFile()
666 }
667 } else if otherName == depNames.CrtEnd {
668 if !c.properties.Nocrt {
669 depPaths.CrtEnd = obj.outputFile()
670 }
671 } else {
672 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.outputFile())
673 }
674 }
675 })
676
677 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800678}
679
Colin Crossed4cf0b2015-03-26 14:43:45 -0700680// ccLinked contains the properties and members used by libraries and executables
681type ccLinked struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800682 ccBase
Colin Crossed4cf0b2015-03-26 14:43:45 -0700683
684 dynamicProperties struct {
685 VariantIsShared bool `blueprint:"mutated"`
686 VariantIsStatic bool `blueprint:"mutated"`
687 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800688}
689
Colin Crossed4cf0b2015-03-26 14:43:45 -0700690func newCCDynamic(dynamic *ccLinked, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700691 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
692
Colin Crossed4cf0b2015-03-26 14:43:45 -0700693 props = append(props, &dynamic.dynamicProperties)
694
Colin Crossc472d572015-03-17 15:06:21 -0700695 return newCCBase(&dynamic.ccBase, module, hod, multilib, props...)
696}
697
Colin Crossed4cf0b2015-03-26 14:43:45 -0700698func (c *ccLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross28d76592015-03-26 16:14:04 -0700699 if ctx.ContainsProperty("system_shared_libs") {
700 return c.properties.System_shared_libs
Colin Cross577f6e42015-03-27 18:23:34 -0700701 } else if ctx.Device() && c.properties.Sdk_version == "" {
702 return []string{"libc", "libm"}
Colin Cross28d76592015-03-26 16:14:04 -0700703 } else {
Colin Cross577f6e42015-03-27 18:23:34 -0700704 return nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800705 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800706}
707
Colin Crossed4cf0b2015-03-26 14:43:45 -0700708func (c *ccLinked) stl(ctx common.AndroidBaseContext) string {
Colin Cross577f6e42015-03-27 18:23:34 -0700709 if c.properties.Sdk_version != "" && ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700710 switch c.properties.Stl {
711 case "":
712 return "ndk_system"
713 case "c++_shared", "c++_static",
714 "stlport_shared", "stlport_static",
715 "gnustl_static":
716 return "ndk_lib" + c.properties.Stl
717 default:
718 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.properties.Stl)
719 return ""
720 }
721 }
722
723 switch c.properties.Stl {
724 case "libc++", "libc++_static",
725 "stlport", "stlport_static",
726 "libstdc++":
727 return c.properties.Stl
728 case "none":
729 return ""
730 case "":
731 if c.shared() {
732 return "libc++" // TODO: mingw needs libstdc++
733 } else {
734 return "libc++_static"
735 }
736 default:
737 ctx.ModuleErrorf("stl: %q is not a supported STL", c.properties.Stl)
738 return ""
739 }
740}
741
742func (c *ccLinked) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
743 stl := c.stl(ctx)
744 if ctx.Failed() {
745 return flags
746 }
747
748 switch stl {
749 case "libc++", "libc++_static":
750 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
751 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/external/libcxx/include")
752 if ctx.Host() {
753 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
754 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
755 flags.LdLibs = append(flags.LdLibs, "-lc", "-lm", "-lpthread")
756 }
757 case "stlport", "stlport_static":
758 if ctx.Device() {
759 flags.IncludeDirs = append(flags.IncludeDirs,
760 "${SrcDir}/external/stlport/stlport",
761 "${SrcDir}/bionic/libstdc++/include",
762 "${SrcDir}/bionic")
763 }
764 case "libstdc++":
765 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
766 // tree is in good enough shape to not need it.
767 // Host builds will use GNU libstdc++.
768 if ctx.Device() {
769 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/bionic/libstdc++/include")
770 }
771 case "ndk_system":
772 ndkSrcRoot := ctx.Config().(Config).SrcDir() + "/prebuilts/ndk/current/sources/"
773 flags.IncludeDirs = append(flags.IncludeDirs, ndkSrcRoot+"cxx-stl/system/include")
774 case "ndk_libc++_shared", "ndk_libc++_static":
775 // TODO(danalbert): This really shouldn't be here...
776 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
777 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
778 // Nothing
779 case "":
780 // None or error.
781 if ctx.Host() {
782 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
783 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
784 flags.LdLibs = append(flags.LdLibs, "-lc", "-lm")
785 }
786 default:
787 panic(fmt.Errorf("Unknown stl in ccLinked.Flags: %q", stl))
788 }
789
790 return flags
791}
792
793func (c *ccLinked) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700794 depNames = c.ccBase.DepNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800795
Colin Crossed4cf0b2015-03-26 14:43:45 -0700796 stl := c.stl(ctx)
797 if ctx.Failed() {
798 return depNames
799 }
800
801 switch stl {
802 case "libc++":
803 depNames.SharedLibs = append(depNames.SharedLibs, stl)
804 case "libstdc++":
805 if ctx.Device() {
806 depNames.SharedLibs = append(depNames.SharedLibs, stl)
807 }
808 case "libc++_static":
809 depNames.StaticLibs = append(depNames.StaticLibs, stl)
810 case "stlport":
811 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
812 case "stlport_static":
813 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
814 case "":
815 // None or error.
816 case "ndk_system":
817 // TODO: Make a system STL prebuilt for the NDK.
818 // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have
819 // its own includes. The includes are handled in ccBase.Flags().
Colin Cross577f6e42015-03-27 18:23:34 -0700820 depNames.SharedLibs = append([]string{"libstdc++"}, depNames.SharedLibs...)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700821 case "ndk_libc++_shared", "ndk_libstlport_shared":
822 depNames.SharedLibs = append(depNames.SharedLibs, stl)
823 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
824 depNames.StaticLibs = append(depNames.StaticLibs, stl)
825 default:
826 panic(fmt.Errorf("Unknown stl in ccLinked.DepNames: %q", stl))
827 }
828
Colin Crossf6566ed2015-03-24 11:13:38 -0700829 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700830 if ctx.ModuleName() != "libcompiler_rt-extras" {
831 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
832 }
Colin Cross77b00fa2015-03-16 16:15:49 -0700833 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700834 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700835
836 if c.shared() {
837 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
838 }
Colin Cross577f6e42015-03-27 18:23:34 -0700839
840 if c.properties.Sdk_version != "" {
841 version := c.properties.Sdk_version
842 depNames.SharedLibs = append(depNames.SharedLibs,
843 "ndk_libc."+version,
844 "ndk_libm."+version,
845 )
846 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800847 }
848
Colin Cross21b9a242015-03-24 14:15:58 -0700849 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800850}
851
Colin Crossed4cf0b2015-03-26 14:43:45 -0700852// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
853type ccLinkedInterface interface {
854 // Returns true if the build options for the module have selected a static or shared build
855 buildStatic() bool
856 buildShared() bool
857
858 // Sets whether a specific variant is static or shared
859 setStatic()
860 setShared()
861
862 // Returns whether a specific variant is static or shared
863 static() bool
864 shared() bool
865}
866
867var _ ccLinkedInterface = (*CCLibrary)(nil)
868var _ ccLinkedInterface = (*CCBinary)(nil)
869
870func (c *ccLinked) static() bool {
871 return c.dynamicProperties.VariantIsStatic
872}
873
874func (c *ccLinked) shared() bool {
875 return c.dynamicProperties.VariantIsShared
876}
877
878func (c *ccLinked) setStatic() {
879 c.dynamicProperties.VariantIsStatic = true
880}
881
882func (c *ccLinked) setShared() {
883 c.dynamicProperties.VariantIsShared = true
884}
885
Colin Cross3f40fa42015-01-30 17:27:36 -0800886type ccExportedIncludeDirsProducer interface {
887 exportedIncludeDirs() []string
888}
889
890//
891// Combined static+shared libraries
892//
893
Colin Cross97ba0732015-03-23 17:50:24 -0700894type CCLibrary struct {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700895 ccLinked
Colin Cross3f40fa42015-01-30 17:27:36 -0800896
Colin Crossed4cf0b2015-03-26 14:43:45 -0700897 reuseFrom ccLibraryInterface
898 reuseObjFiles []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800899 objFiles []string
900 exportIncludeDirs []string
901 out string
902
Colin Cross97ba0732015-03-23 17:50:24 -0700903 LibraryProperties struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800904 BuildStatic bool `blueprint:"mutated"`
905 BuildShared bool `blueprint:"mutated"`
Colin Crossed4cf0b2015-03-26 14:43:45 -0700906 Static struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800907 Srcs []string `android:"arch_variant"`
908 Cflags []string `android:"arch_variant"`
909 } `android:"arch_variant"`
910 Shared struct {
911 Srcs []string `android:"arch_variant"`
912 Cflags []string `android:"arch_variant"`
913 } `android:"arch_variant"`
914 }
915}
916
Colin Crossed4cf0b2015-03-26 14:43:45 -0700917func (c *CCLibrary) buildStatic() bool {
918 return c.LibraryProperties.BuildStatic
919}
920
921func (c *CCLibrary) buildShared() bool {
922 return c.LibraryProperties.BuildShared
923}
924
Colin Cross97ba0732015-03-23 17:50:24 -0700925type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700926 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -0700927 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700928 setReuseFrom(ccLibraryInterface)
929 getReuseFrom() ccLibraryInterface
930 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -0700931 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -0700932}
933
Colin Crossed4cf0b2015-03-26 14:43:45 -0700934var _ ccLibraryInterface = (*CCLibrary)(nil)
935
Colin Cross97ba0732015-03-23 17:50:24 -0700936func (c *CCLibrary) ccLibrary() *CCLibrary {
937 return c
Colin Cross3f40fa42015-01-30 17:27:36 -0800938}
939
Colin Cross97ba0732015-03-23 17:50:24 -0700940func NewCCLibrary(library *CCLibrary, module CCModuleType,
941 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
942
Colin Crossed4cf0b2015-03-26 14:43:45 -0700943 return newCCDynamic(&library.ccLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -0700944 &library.LibraryProperties)
945}
946
947func CCLibraryFactory() (blueprint.Module, []interface{}) {
948 module := &CCLibrary{}
949
950 module.LibraryProperties.BuildShared = true
951 module.LibraryProperties.BuildStatic = true
952
953 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
954}
955
Colin Cross21b9a242015-03-24 14:15:58 -0700956func (c *CCLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700957 depNames = c.ccLinked.DepNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -0700958 if c.shared() {
Colin Crossf6566ed2015-03-24 11:13:38 -0700959 if ctx.Device() {
Colin Cross21b9a242015-03-24 14:15:58 -0700960 depNames.CrtBegin = "crtbegin_so"
961 depNames.CrtEnd = "crtend_so"
Colin Cross3f40fa42015-01-30 17:27:36 -0800962 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800963 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800964
Colin Cross21b9a242015-03-24 14:15:58 -0700965 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800966}
967
Colin Cross97ba0732015-03-23 17:50:24 -0700968func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800969 return c.out
970}
971
Colin Crossed4cf0b2015-03-26 14:43:45 -0700972func (c *CCLibrary) getReuseObjFiles() []string {
973 return c.reuseObjFiles
974}
975
976func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
977 c.reuseFrom = reuseFrom
978}
979
980func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
981 return c.reuseFrom
982}
983
Colin Cross97ba0732015-03-23 17:50:24 -0700984func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800985 return c.objFiles
986}
987
Colin Cross97ba0732015-03-23 17:50:24 -0700988func (c *CCLibrary) exportedIncludeDirs() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800989 return c.exportIncludeDirs
990}
991
Colin Cross21b9a242015-03-24 14:15:58 -0700992func (c *CCLibrary) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700993 flags = c.ccLinked.Flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -0700994
Colin Cross97ba0732015-03-23 17:50:24 -0700995 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -0800996
Colin Crossed4cf0b2015-03-26 14:43:45 -0700997 if c.shared() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800998 libName := ctx.ModuleName()
999 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
1000 sharedFlag := "-Wl,-shared"
Colin Crossf6566ed2015-03-24 11:13:38 -07001001 if c.properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001002 sharedFlag = "-shared"
1003 }
Colin Crossf6566ed2015-03-24 11:13:38 -07001004 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001005 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001006 }
Colin Cross97ba0732015-03-23 17:50:24 -07001007
1008 flags.LdFlags = append(flags.LdFlags,
1009 "-Wl,--gc-sections",
1010 sharedFlag,
1011 "-Wl,-soname,"+libName+sharedLibraryExtension,
1012 )
Colin Cross3f40fa42015-01-30 17:27:36 -08001013 }
Colin Cross97ba0732015-03-23 17:50:24 -07001014
1015 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001016}
1017
Colin Cross97ba0732015-03-23 17:50:24 -07001018func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1019 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001020
1021 staticFlags := flags
Colin Cross97ba0732015-03-23 17:50:24 -07001022 staticFlags.CFlags = append(staticFlags.CFlags, c.LibraryProperties.Static.Cflags...)
Colin Cross581c1892015-04-07 16:50:10 -07001023 objFilesStatic := c.customCompileObjs(ctx, staticFlags, common.DeviceStaticLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001024 c.LibraryProperties.Static.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001025
1026 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001027 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001028
1029 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1030
1031 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1032
1033 c.objFiles = objFiles
1034 c.out = outputFile
1035 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1036 common.ModuleSrcDir(ctx))
1037
1038 ctx.CheckbuildFile(outputFile)
1039}
1040
Colin Cross97ba0732015-03-23 17:50:24 -07001041func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1042 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001043
1044 sharedFlags := flags
Colin Cross97ba0732015-03-23 17:50:24 -07001045 sharedFlags.CFlags = append(sharedFlags.CFlags, c.LibraryProperties.Shared.Cflags...)
Colin Cross581c1892015-04-07 16:50:10 -07001046 objFilesShared := c.customCompileObjs(ctx, sharedFlags, common.DeviceSharedLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001047 c.LibraryProperties.Shared.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001048
1049 objFiles = append(objFiles, objFilesShared...)
1050
1051 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1052
Colin Cross97ba0732015-03-23 17:50:24 -07001053 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001054 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, false,
Colin Cross77b00fa2015-03-16 16:15:49 -07001055 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001056
1057 c.out = outputFile
1058 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1059 common.ModuleSrcDir(ctx))
Colin Cross3f40fa42015-01-30 17:27:36 -08001060}
1061
Colin Cross97ba0732015-03-23 17:50:24 -07001062func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1063 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001064
1065 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001066 if c.getReuseFrom().ccLibrary() == c {
1067 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001068 } else {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001069 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001070 }
1071
Colin Crossed4cf0b2015-03-26 14:43:45 -07001072 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001073 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1074 } else {
1075 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1076 }
1077}
1078
Colin Cross97ba0732015-03-23 17:50:24 -07001079func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001080 // Static libraries do not get installed.
1081}
1082
Colin Cross97ba0732015-03-23 17:50:24 -07001083func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001084 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001085 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001086 installDir = "lib64"
1087 }
1088
1089 ctx.InstallFile(installDir, c.out)
1090}
1091
Colin Cross97ba0732015-03-23 17:50:24 -07001092func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001093 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001094 c.installStaticLibrary(ctx, flags)
1095 } else {
1096 c.installSharedLibrary(ctx, flags)
1097 }
1098}
1099
Colin Cross3f40fa42015-01-30 17:27:36 -08001100//
1101// Objects (for crt*.o)
1102//
1103
1104type ccObject struct {
1105 ccBase
1106 out string
1107}
1108
Colin Cross97ba0732015-03-23 17:50:24 -07001109func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001110 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001111
Colin Crossc472d572015-03-17 15:06:21 -07001112 return newCCBase(&module.ccBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001113}
1114
1115func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1116 // object files can't have any dynamic dependencies
1117 return nil
1118}
1119
Colin Cross21b9a242015-03-24 14:15:58 -07001120func (*ccObject) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1121 // object files can't have any dynamic dependencies
1122 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001123}
1124
1125func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001126 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001127
Colin Cross97ba0732015-03-23 17:50:24 -07001128 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001129
1130 var outputFile string
1131 if len(objFiles) == 1 {
1132 outputFile = objFiles[0]
1133 } else {
1134 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+".o")
1135 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1136 }
1137
1138 c.out = outputFile
1139
1140 ctx.CheckbuildFile(outputFile)
1141}
1142
Colin Cross97ba0732015-03-23 17:50:24 -07001143func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001144 // Object files do not get installed.
1145}
1146
Colin Cross3f40fa42015-01-30 17:27:36 -08001147func (c *ccObject) outputFile() string {
1148 return c.out
1149}
1150
1151//
1152// Executables
1153//
1154
Colin Cross97ba0732015-03-23 17:50:24 -07001155type CCBinary struct {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001156 ccLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001157 out string
Colin Cross97ba0732015-03-23 17:50:24 -07001158 BinaryProperties struct {
1159 // static_executable: compile executable with -static
1160 Static_executable bool
1161
1162 // stem: set the name of the output
1163 Stem string `android:"arch_variant"`
1164
Colin Cross4ae185c2015-03-26 15:12:10 -07001165 // suffix: append to the name of the output
1166 Suffix string `android:"arch_variant"`
1167
Colin Cross97ba0732015-03-23 17:50:24 -07001168 // prefix_symbols: if set, add an extra objcopy --prefix-symbols= step
1169 Prefix_symbols string
1170 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001171}
1172
Colin Crossed4cf0b2015-03-26 14:43:45 -07001173func (c *CCBinary) buildStatic() bool {
1174 return c.BinaryProperties.Static_executable
1175}
1176
1177func (c *CCBinary) buildShared() bool {
1178 return !c.BinaryProperties.Static_executable
1179}
1180
Colin Cross97ba0732015-03-23 17:50:24 -07001181func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001182 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001183 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001184 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001185 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001186
1187 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001188}
1189
Colin Cross21b9a242015-03-24 14:15:58 -07001190func (c *CCBinary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001191 depNames = c.ccLinked.DepNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001192 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001193 if c.BinaryProperties.Static_executable {
Colin Cross21b9a242015-03-24 14:15:58 -07001194 depNames.CrtBegin = "crtbegin_static"
Colin Cross3f40fa42015-01-30 17:27:36 -08001195 } else {
Colin Cross21b9a242015-03-24 14:15:58 -07001196 depNames.CrtBegin = "crtbegin_dynamic"
Colin Cross3f40fa42015-01-30 17:27:36 -08001197 }
Colin Cross21b9a242015-03-24 14:15:58 -07001198 depNames.CrtEnd = "crtend_android"
Colin Crossed4cf0b2015-03-26 14:43:45 -07001199
1200 if c.BinaryProperties.Static_executable {
1201 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1202 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1203 // move them to the beginning of deps.LateStaticLibs
1204 var groupLibs []string
1205 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1206 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1207 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1208 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001209 }
Colin Cross21b9a242015-03-24 14:15:58 -07001210 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001211}
1212
Colin Cross97ba0732015-03-23 17:50:24 -07001213func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001214 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001215
Colin Cross1f8f2342015-03-26 16:09:47 -07001216 props = append(props, &binary.BinaryProperties)
1217
1218 return newCCDynamic(&binary.ccLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001219}
1220
Colin Cross97ba0732015-03-23 17:50:24 -07001221func CCBinaryFactory() (blueprint.Module, []interface{}) {
1222 module := &CCBinary{}
1223
1224 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001225}
1226
Colin Cross21b9a242015-03-24 14:15:58 -07001227func (c *CCBinary) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001228 flags = c.ccLinked.Flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001229
Colin Cross97ba0732015-03-23 17:50:24 -07001230 flags.CFlags = append(flags.CFlags, "-fpie")
1231
Colin Crossf6566ed2015-03-24 11:13:38 -07001232 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001233 if c.BinaryProperties.Static_executable {
1234 // Clang driver needs -static to create static executable.
1235 // However, bionic/linker uses -shared to overwrite.
1236 // Linker for x86 targets does not allow coexistance of -static and -shared,
1237 // so we add -static only if -shared is not used.
1238 if !inList("-shared", flags.LdFlags) {
1239 flags.LdFlags = append(flags.LdFlags, "-static")
1240 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001241
Colin Crossed4cf0b2015-03-26 14:43:45 -07001242 flags.LdFlags = append(flags.LdFlags,
1243 "-nostdlib",
1244 "-Bstatic",
1245 "-Wl,--gc-sections",
1246 )
1247
1248 } else {
1249 linker := "/system/bin/linker"
1250 if flags.Toolchain.Is64Bit() {
1251 linker = "/system/bin/linker64"
1252 }
1253
1254 flags.LdFlags = append(flags.LdFlags,
1255 "-nostdlib",
1256 "-Bdynamic",
1257 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1258 "-Wl,--gc-sections",
1259 "-Wl,-z,nocopyreloc",
1260 )
1261 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001262 }
1263
Colin Cross97ba0732015-03-23 17:50:24 -07001264 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001265}
1266
Colin Cross97ba0732015-03-23 17:50:24 -07001267func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1268 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001269
Colin Cross97ba0732015-03-23 17:50:24 -07001270 if !c.BinaryProperties.Static_executable && inList("libc", c.properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001271 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1272 "from static libs or set static_executable: true")
1273 }
1274
1275 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001276 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001277 if c.BinaryProperties.Prefix_symbols != "" {
1278 afterPrefixSymbols := outputFile
1279 outputFile = outputFile + ".intermediate"
1280 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1281 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1282 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001283
Colin Cross97ba0732015-03-23 17:50:24 -07001284 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001285 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001286 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001287}
Colin Cross3f40fa42015-01-30 17:27:36 -08001288
Colin Cross97ba0732015-03-23 17:50:24 -07001289func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001290 ctx.InstallFile("bin", c.out)
1291}
1292
1293type ccTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001294 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001295
1296 testProperties struct {
1297 // test_per_src: Create a separate test for each source file. Useful when there is
1298 // global state that can not be torn down and reset between each test suite.
1299 Test_per_src bool
1300 }
Dan Albertc403f7c2015-03-18 14:01:18 -07001301}
1302
Colin Cross21b9a242015-03-24 14:15:58 -07001303func (c *ccTest) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
1304 flags = c.CCBinary.Flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001305
Colin Cross97ba0732015-03-23 17:50:24 -07001306 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001307 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001308 flags.CFlags = append(flags.CFlags, "-O0", "-g")
1309 flags.LdLibs = append(flags.LdLibs, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001310 }
1311
1312 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross97ba0732015-03-23 17:50:24 -07001313 flags.IncludeDirs = append(flags.IncludeDirs,
Tim Kilbourn5ccc7302015-03-19 10:02:21 -07001314 filepath.Join(ctx.Config().(Config).SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001315
Colin Cross21b9a242015-03-24 14:15:58 -07001316 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001317}
1318
Colin Cross21b9a242015-03-24 14:15:58 -07001319func (c *ccTest) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1320 depNames = c.CCBinary.DepNames(ctx, depNames)
1321 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
1322 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001323}
1324
Colin Cross97ba0732015-03-23 17:50:24 -07001325func (c *ccTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001326 if ctx.Device() {
Tim Kilbourn5ccc7302015-03-19 10:02:21 -07001327 ctx.InstallFile("../data/nativetest/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001328 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001329 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001330 }
1331}
1332
Colin Cross97ba0732015-03-23 17:50:24 -07001333func CCTestFactory() (blueprint.Module, []interface{}) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001334 module := &ccTest{}
Colin Cross1f8f2342015-03-26 16:09:47 -07001335 return NewCCBinary(&module.CCBinary, module, common.HostAndDeviceSupported,
1336 &module.testProperties)
Colin Cross6b290692015-03-19 14:05:33 -07001337}
1338
1339func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
1340 if test, ok := mctx.Module().(*ccTest); ok {
1341 if test.testProperties.Test_per_src {
1342 testNames := make([]string, len(test.properties.Srcs))
1343 for i, src := range test.properties.Srcs {
1344 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1345 }
1346 tests := mctx.CreateLocalVariations(testNames...)
1347 for i, src := range test.properties.Srcs {
1348 tests[i].(*ccTest).properties.Srcs = []string{src}
Colin Cross97ba0732015-03-23 17:50:24 -07001349 tests[i].(*ccTest).BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001350 }
1351 }
1352 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001353}
1354
1355//
1356// Static library
1357//
1358
Colin Cross97ba0732015-03-23 17:50:24 -07001359func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1360 module := &CCLibrary{}
1361 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001362
Colin Cross97ba0732015-03-23 17:50:24 -07001363 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001364}
1365
1366//
1367// Shared libraries
1368//
1369
Colin Cross97ba0732015-03-23 17:50:24 -07001370func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1371 module := &CCLibrary{}
1372 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001373
Colin Cross97ba0732015-03-23 17:50:24 -07001374 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001375}
1376
1377//
1378// Host static library
1379//
1380
Colin Cross97ba0732015-03-23 17:50:24 -07001381func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1382 module := &CCLibrary{}
1383 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001384
Colin Cross97ba0732015-03-23 17:50:24 -07001385 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001386}
1387
1388//
1389// Host Shared libraries
1390//
1391
Colin Cross97ba0732015-03-23 17:50:24 -07001392func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1393 module := &CCLibrary{}
1394 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001395
Colin Cross97ba0732015-03-23 17:50:24 -07001396 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001397}
1398
1399//
1400// Host Binaries
1401//
1402
Colin Cross97ba0732015-03-23 17:50:24 -07001403func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1404 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001405
Colin Cross97ba0732015-03-23 17:50:24 -07001406 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001407}
1408
1409//
Colin Cross1f8f2342015-03-26 16:09:47 -07001410// Host Tests
1411//
1412
1413func CCTestHostFactory() (blueprint.Module, []interface{}) {
1414 module := &ccTest{}
1415 return NewCCBinary(&module.CCBinary, module, common.HostSupported,
1416 &module.testProperties)
1417}
1418
1419//
Colin Cross3f40fa42015-01-30 17:27:36 -08001420// Device libraries shipped with gcc
1421//
1422
1423type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001424 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001425}
1426
1427func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1428 // toolchain libraries can't have any dependencies
1429 return nil
1430}
1431
Colin Cross21b9a242015-03-24 14:15:58 -07001432func (*toolchainLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001433 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001434 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001435}
1436
Colin Cross97ba0732015-03-23 17:50:24 -07001437func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001438 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001439
Colin Cross97ba0732015-03-23 17:50:24 -07001440 module.LibraryProperties.BuildStatic = true
1441
Colin Cross21b9a242015-03-24 14:15:58 -07001442 return newCCBase(&module.ccBase, module, common.DeviceSupported, common.MultilibBoth,
1443 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001444}
1445
1446func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001447 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001448
1449 libName := ctx.ModuleName() + staticLibraryExtension
1450 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1451
1452 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1453
1454 c.out = outputFile
1455
1456 ctx.CheckbuildFile(outputFile)
1457}
1458
Colin Cross97ba0732015-03-23 17:50:24 -07001459func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001460 // Toolchain libraries do not get installed.
1461}
1462
Dan Albertbe961682015-03-18 23:38:50 -07001463// NDK prebuilt libraries.
1464//
1465// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1466// either (with the exception of the shared STLs, which are installed to the app's directory rather
1467// than to the system image).
1468
1469func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1470 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
1471 ctx.Config().(Config).SrcDir(), version, toolchain.Name())
1472}
1473
1474type ndkPrebuiltLibrary struct {
1475 CCLibrary
1476}
1477
1478func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1479 ctx common.AndroidDynamicDependerModuleContext) []string {
1480
1481 // NDK libraries can't have any dependencies
1482 return nil
1483}
1484
1485func (*ndkPrebuiltLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1486 // NDK libraries can't have any dependencies
1487 return CCDeps{}
1488}
1489
1490func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1491 module := &ndkPrebuiltLibrary{}
1492 module.LibraryProperties.BuildShared = true
1493 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1494}
1495
1496func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1497 deps CCDeps, objFiles []string) {
1498 // A null build step, but it sets up the output path.
1499 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1500 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1501 }
1502
1503 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1504 common.ModuleSrcDir(ctx))
1505
1506 // NDK prebuilt libraries are named like: ndk_LIBNAME.SDK_VERSION.
1507 // We want to translate to just LIBNAME.
1508 libName := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1509 libDir := getNdkLibDir(ctx, flags.Toolchain, c.properties.Sdk_version)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001510 c.out = filepath.Join(libDir, libName+sharedLibraryExtension)
Dan Albertbe961682015-03-18 23:38:50 -07001511}
1512
1513func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1514 // Toolchain libraries do not get installed.
1515}
1516
1517// The NDK STLs are slightly different from the prebuilt system libraries:
1518// * Are not specific to each platform version.
1519// * The libraries are not in a predictable location for each STL.
1520
1521type ndkPrebuiltStl struct {
1522 ndkPrebuiltLibrary
1523}
1524
1525type ndkPrebuiltStaticStl struct {
1526 ndkPrebuiltStl
1527}
1528
1529type ndkPrebuiltSharedStl struct {
1530 ndkPrebuiltStl
1531}
1532
1533func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1534 module := &ndkPrebuiltSharedStl{}
1535 module.LibraryProperties.BuildShared = true
1536 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1537}
1538
1539func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1540 module := &ndkPrebuiltStaticStl{}
1541 module.LibraryProperties.BuildStatic = true
1542 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1543}
1544
1545func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1546 gccVersion := toolchain.GccVersion()
1547 var libDir string
1548 switch stl {
1549 case "libstlport":
1550 libDir = "cxx-stl/stlport/libs"
1551 case "libc++":
1552 libDir = "cxx-stl/llvm-libc++/libs"
1553 case "libgnustl":
1554 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1555 }
1556
1557 if libDir != "" {
1558 ndkSrcRoot := ctx.Config().(Config).SrcDir() + "/prebuilts/ndk/current/sources"
1559 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1560 }
1561
1562 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1563 return ""
1564}
1565
1566func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1567 deps CCDeps, objFiles []string) {
1568 // A null build step, but it sets up the output path.
1569 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1570 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1571 }
1572
1573 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1574 common.ModuleSrcDir(ctx))
1575
1576 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1577 libExt := sharedLibraryExtension
1578 if c.LibraryProperties.BuildStatic {
1579 libExt = staticLibraryExtension
1580 }
1581
1582 stlName := strings.TrimSuffix(libName, "_shared")
1583 stlName = strings.TrimSuffix(stlName, "_static")
1584 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1585 c.out = libDir + "/" + libName + libExt
1586}
1587
Colin Cross3f40fa42015-01-30 17:27:36 -08001588func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001589 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001590 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001591 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001592 modules = mctx.CreateLocalVariations("static", "shared")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001593 modules[0].(ccLinkedInterface).setStatic()
1594 modules[1].(ccLinkedInterface).setShared()
1595 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001596 modules = mctx.CreateLocalVariations("static")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001597 modules[0].(ccLinkedInterface).setStatic()
1598 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001599 modules = mctx.CreateLocalVariations("shared")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001600 modules[0].(ccLinkedInterface).setShared()
Colin Cross3f40fa42015-01-30 17:27:36 -08001601 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001602 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001603 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001604
1605 if _, ok := c.(ccLibraryInterface); ok {
1606 reuseFrom := modules[0].(ccLibraryInterface)
1607 for _, m := range modules {
1608 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001609 }
1610 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001611 }
1612}