blob: d400df08810c4fc214ad7f3eb66679b58baa87e8 [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
35 PrebuiltOS() string
36}
37
38var (
39 HostPrebuiltTag = pctx.VariableConfigMethod("HostPrebuiltTag", Config.PrebuiltOS)
40 SrcDir = pctx.VariableConfigMethod("SrcDir", Config.SrcDir)
41
42 LibcRoot = pctx.StaticVariable("LibcRoot", "${SrcDir}/bionic/libc")
43 LibmRoot = pctx.StaticVariable("LibmRoot", "${SrcDir}/bionic/libm")
44)
45
46// Flags used by lots of devices. Putting them in package static variables will save bytes in
47// build.ninja so they aren't repeated for every file
48var (
49 commonGlobalCflags = []string{
50 "-DANDROID",
51 "-fmessage-length=0",
52 "-W",
53 "-Wall",
54 "-Wno-unused",
55 "-Winit-self",
56 "-Wpointer-arith",
57
58 // COMMON_RELEASE_CFLAGS
59 "-DNDEBUG",
60 "-UDEBUG",
61 }
62
63 deviceGlobalCflags = []string{
64 // TARGET_ERROR_FLAGS
65 "-Werror=return-type",
66 "-Werror=non-virtual-dtor",
67 "-Werror=address",
68 "-Werror=sequence-point",
69 }
70
71 hostGlobalCflags = []string{}
72
73 commonGlobalCppflags = []string{
74 "-Wsign-promo",
75 "-std=gnu++11",
76 }
77)
78
79func init() {
80 pctx.StaticVariable("commonGlobalCflags", strings.Join(commonGlobalCflags, " "))
81 pctx.StaticVariable("deviceGlobalCflags", strings.Join(deviceGlobalCflags, " "))
82 pctx.StaticVariable("hostGlobalCflags", strings.Join(hostGlobalCflags, " "))
83
84 pctx.StaticVariable("commonGlobalCppflags", strings.Join(commonGlobalCppflags, " "))
85
86 pctx.StaticVariable("commonClangGlobalCflags",
87 strings.Join(clangFilterUnknownCflags(commonGlobalCflags), " "))
88 pctx.StaticVariable("deviceClangGlobalCflags",
89 strings.Join(clangFilterUnknownCflags(deviceGlobalCflags), " "))
90 pctx.StaticVariable("hostClangGlobalCflags",
91 strings.Join(clangFilterUnknownCflags(hostGlobalCflags), " "))
Tim Kilbournf2948142015-03-11 12:03:03 -070092 pctx.StaticVariable("commonClangGlobalCppflags",
93 strings.Join(clangFilterUnknownCflags(commonGlobalCppflags), " "))
Colin Cross3f40fa42015-01-30 17:27:36 -080094
95 // Everything in this list is a crime against abstraction and dependency tracking.
96 // Do not add anything to this list.
97 pctx.StaticVariable("commonGlobalIncludes", strings.Join([]string{
98 "-isystem ${SrcDir}/system/core/include",
99 "-isystem ${SrcDir}/hardware/libhardware/include",
100 "-isystem ${SrcDir}/hardware/libhardware_legacy/include",
101 "-isystem ${SrcDir}/hardware/ril/include",
102 "-isystem ${SrcDir}/libnativehelper/include",
103 "-isystem ${SrcDir}/frameworks/native/include",
104 "-isystem ${SrcDir}/frameworks/native/opengl/include",
105 "-isystem ${SrcDir}/frameworks/av/include",
106 "-isystem ${SrcDir}/frameworks/base/include",
107 }, " "))
108
109 pctx.StaticVariable("clangPath", "${SrcDir}/prebuilts/clang/${HostPrebuiltTag}/host/3.6/bin/")
110}
111
Colin Cross97ba0732015-03-23 17:50:24 -0700112// ccProperties describes properties used to compile all C or C++ modules
Colin Cross3f40fa42015-01-30 17:27:36 -0800113type ccProperties struct {
114 // srcs: list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
115 Srcs []string `android:"arch_variant,arch_subtract"`
116
117 // cflags: list of module-specific flags that will be used for C and C++ compiles.
118 Cflags []string `android:"arch_variant"`
119
120 // cppflags: list of module-specific flags that will be used for C++ compiles
121 Cppflags []string `android:"arch_variant"`
122
123 // conlyflags: list of module-specific flags that will be used for C compiles
124 Conlyflags []string `android:"arch_variant"`
125
126 // asflags: list of module-specific flags that will be used for .S compiles
127 Asflags []string `android:"arch_variant"`
128
129 // ldflags: list of module-specific flags that will be used for all link steps
130 Ldflags []string `android:"arch_variant"`
131
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700132 // instruction_set: the instruction set architecture to use to compile the C/C++
133 // module.
134 Instruction_set string `android:"arch_variant"`
135
Colin Cross3f40fa42015-01-30 17:27:36 -0800136 // include_dirs: list of directories relative to the root of the source tree that will
137 // be added to the include path using -I.
138 // If possible, don't use this. If adding paths from the current directory use
139 // local_include_dirs, if adding paths from other modules use export_include_dirs in
140 // that module.
141 Include_dirs []string `android:"arch_variant"`
142
143 // local_include_dirs: list of directories relative to the Blueprints file that will
144 // be added to the include path using -I
145 Local_include_dirs []string `android:"arch_variant"`
146
147 // export_include_dirs: list of directories relative to the Blueprints file that will
148 // be added to the include path using -I for any module that links against this module
Dan Albertbe961682015-03-18 23:38:50 -0700149 Export_include_dirs []string `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800150
151 // clang_cflags: list of module-specific flags that will be used for C and C++ compiles when
152 // compiling with clang
153 Clang_cflags []string `android:"arch_variant"`
154
155 // clang_asflags: list of module-specific flags that will be used for .S compiles when
156 // compiling with clang
157 Clang_asflags []string `android:"arch_variant"`
158
159 // system_shared_libs: list of system libraries that will be dynamically linked to
160 // shared library and executable modules. If unset, generally defaults to libc
161 // and libm. Set to [] to prevent linking against libc and libm.
162 System_shared_libs []string
163
164 // whole_static_libs: list of modules whose object files should be linked into this module
165 // in their entirety. For static library modules, all of the .o files from the intermediate
166 // directory of the dependency will be linked into this modules .a file. For a shared library,
167 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
168 Whole_static_libs []string `android:"arch_variant"`
169
170 // static_libs: list of modules that should be statically linked into this module.
171 Static_libs []string `android:"arch_variant"`
172
173 // shared_libs: list of modules that should be dynamically linked into this module.
174 Shared_libs []string `android:"arch_variant"`
175
176 // allow_undefined_symbols: allow the module to contain undefined symbols. By default,
177 // modules cannot contain undefined symbols that are not satisified by their immediate
178 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
179 // This flag should only be necessary for compiling low-level libraries like libc.
180 Allow_undefined_symbols bool
181
182 // nocrt: don't link in crt_begin and crt_end. This flag should only be necessary for
183 // compiling crt or libc.
184 Nocrt bool `android:"arch_variant"`
185
186 // no_default_compiler_flags: don't insert default compiler flags into asflags, cflags,
187 // cppflags, conlyflags, ldflags, or include_dirs
188 No_default_compiler_flags bool
189
190 // clang: compile module with clang instead of gcc
191 Clang bool `android:"arch_variant"`
192
193 // rtti: pass -frtti instead of -fno-rtti
194 Rtti bool
195
196 // host_ldlibs: -l arguments to pass to linker for host-provided shared libraries
197 Host_ldlibs []string `android:"arch_variant"`
198
199 // stl: select the STL library to use. Possible values are "libc++", "libc++_static",
200 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
201 // default
202 Stl string
203
204 // Set for combined shared/static libraries to prevent compiling object files a second time
205 SkipCompileObjs bool `blueprint:"mutated"`
Colin Crossaf19a292015-03-18 12:07:10 -0700206
207 Debug struct {
208 Cflags []string `android:"arch_variant"`
209 } `android:"arch_variant"`
210 Release struct {
211 Cflags []string `android:"arch_variant"`
212 } `android:"arch_variant"`
Colin Crossefd8e482015-03-18 17:17:35 -0700213
214 // Minimum sdk version supported when compiling against the ndk
215 Sdk_version string
Colin Cross3f40fa42015-01-30 17:27:36 -0800216}
217
218type unusedProperties struct {
219 Asan bool
220 Native_coverage bool
221 Strip string
222 Tags []string
223 Required []string
224}
225
226// Building C/C++ code is handled by objects that satisfy this interface via composition
Colin Cross97ba0732015-03-23 17:50:24 -0700227type CCModuleType interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800228 common.AndroidModule
229
Colin Cross21b9a242015-03-24 14:15:58 -0700230 // Modify the ccFlags
231 Flags(common.AndroidModuleContext, CCFlags) CCFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800232
Colin Cross21b9a242015-03-24 14:15:58 -0700233 // Return list of dependency names for use in AndroidDynamicDependencies and in depsToPaths
234 DepNames(common.AndroidBaseContext, CCDeps) CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800235
236 // Compile objects into final module
Colin Cross97ba0732015-03-23 17:50:24 -0700237 compileModule(common.AndroidModuleContext, CCFlags, CCDeps, []string)
Colin Cross3f40fa42015-01-30 17:27:36 -0800238
Dan Albertc403f7c2015-03-18 14:01:18 -0700239 // Install the built module.
Colin Cross97ba0732015-03-23 17:50:24 -0700240 installModule(common.AndroidModuleContext, CCFlags)
Dan Albertc403f7c2015-03-18 14:01:18 -0700241
Colin Cross3f40fa42015-01-30 17:27:36 -0800242 // Return the output file (.o, .a or .so) for use by other modules
243 outputFile() string
244}
245
Colin Cross97ba0732015-03-23 17:50:24 -0700246type CCDeps struct {
247 StaticLibs, SharedLibs, LateStaticLibs, WholeStaticLibs, ObjFiles, IncludeDirs []string
Colin Crossc472d572015-03-17 15:06:21 -0700248
Colin Cross21b9a242015-03-24 14:15:58 -0700249 WholeStaticLibObjFiles []string
250
Colin Cross97ba0732015-03-23 17:50:24 -0700251 CrtBegin, CrtEnd string
Colin Crossc472d572015-03-17 15:06:21 -0700252}
253
Colin Cross97ba0732015-03-23 17:50:24 -0700254type CCFlags struct {
255 GlobalFlags []string
256 AsFlags []string
257 CFlags []string
258 ConlyFlags []string
259 CppFlags []string
260 LdFlags []string
261 LdLibs []string
262 IncludeDirs []string
263 Nocrt bool
264 Toolchain Toolchain
265 Clang bool
Colin Crossc472d572015-03-17 15:06:21 -0700266}
267
268// ccBase contains the properties and members used by all C/C++ module types, and implements
269// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
270// and uses a ccModuleType interface to that struct to create the build steps.
271type ccBase struct {
272 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700273 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700274
275 properties ccProperties
276 unused unusedProperties
277
278 installPath string
279}
280
Colin Cross97ba0732015-03-23 17:50:24 -0700281func newCCBase(base *ccBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700282 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
283
284 base.module = module
285
286 props = append(props, &base.properties, &base.unused)
287
Colin Cross5049f022015-03-18 13:28:46 -0700288 return common.InitAndroidArchModule(module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700289}
290
Colin Cross3f40fa42015-01-30 17:27:36 -0800291func (c *ccBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
292 toolchain := c.findToolchain(ctx)
293 if ctx.Failed() {
294 return
295 }
296
Colin Cross21b9a242015-03-24 14:15:58 -0700297 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800298 if ctx.Failed() {
299 return
300 }
301
Colin Cross21b9a242015-03-24 14:15:58 -0700302 depNames := c.module.DepNames(ctx, CCDeps{})
Colin Cross3f40fa42015-01-30 17:27:36 -0800303 if ctx.Failed() {
304 return
305 }
306
Colin Cross21b9a242015-03-24 14:15:58 -0700307 deps := c.depsToPaths(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800308 if ctx.Failed() {
309 return
310 }
311
Colin Cross97ba0732015-03-23 17:50:24 -0700312 flags.IncludeDirs = append(flags.IncludeDirs, deps.IncludeDirs...)
Colin Crossed9f8682015-03-18 17:17:35 -0700313
Colin Cross3f40fa42015-01-30 17:27:36 -0800314 objFiles := c.compileObjs(ctx, flags, deps)
315 if ctx.Failed() {
316 return
317 }
318
Colin Cross5049f022015-03-18 13:28:46 -0700319 generatedObjFiles := c.compileGeneratedObjs(ctx, flags, deps)
320 if ctx.Failed() {
321 return
322 }
323
324 objFiles = append(objFiles, generatedObjFiles...)
325
Colin Cross3f40fa42015-01-30 17:27:36 -0800326 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
327 if ctx.Failed() {
328 return
329 }
Dan Albertc403f7c2015-03-18 14:01:18 -0700330
331 c.ccModuleType().installModule(ctx, flags)
332 if ctx.Failed() {
333 return
334 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800335}
336
Colin Cross97ba0732015-03-23 17:50:24 -0700337func (c *ccBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800338 return c.module
339}
340
341var _ common.AndroidDynamicDepender = (*ccBase)(nil)
342
Colin Cross97ba0732015-03-23 17:50:24 -0700343func (c *ccBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800344 arch := ctx.Arch()
345 factory := toolchainFactories[arch.HostOrDevice][arch.ArchType]
346 if factory == nil {
347 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
348 arch.HostOrDevice.String(), arch.String()))
349 }
350 return factory(arch.ArchVariant, arch.CpuVariant)
351}
352
Colin Cross21b9a242015-03-24 14:15:58 -0700353func (c *ccBase) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
354 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.properties.Whole_static_libs...)
355 depNames.StaticLibs = append(depNames.StaticLibs, c.properties.Static_libs...)
356 depNames.SharedLibs = append(depNames.SharedLibs, c.properties.Shared_libs...)
357
Colin Cross21b9a242015-03-24 14:15:58 -0700358 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800359}
360
361func (c *ccBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
Colin Cross21b9a242015-03-24 14:15:58 -0700362 depNames := CCDeps{}
363 depNames = c.module.DepNames(ctx, depNames)
364 staticLibs := depNames.WholeStaticLibs
365 staticLibs = append(staticLibs, depNames.StaticLibs...)
366 staticLibs = append(staticLibs, depNames.LateStaticLibs...)
367 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800368
Colin Cross21b9a242015-03-24 14:15:58 -0700369 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, depNames.SharedLibs...)
370
371 ret := append([]string(nil), depNames.ObjFiles...)
372 if depNames.CrtBegin != "" {
373 ret = append(ret, depNames.CrtBegin)
374 }
375 if depNames.CrtEnd != "" {
376 ret = append(ret, depNames.CrtEnd)
377 }
378
379 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800380}
381
382// Create a ccFlags struct that collects the compile flags from global values,
383// per-target values, module type values, and per-module Blueprints properties
Colin Cross21b9a242015-03-24 14:15:58 -0700384func (c *ccBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700385 flags := CCFlags{
386 CFlags: c.properties.Cflags,
387 CppFlags: c.properties.Cppflags,
388 ConlyFlags: c.properties.Conlyflags,
389 LdFlags: c.properties.Ldflags,
390 AsFlags: c.properties.Asflags,
391 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 Cross97ba0732015-03-23 17:50:24 -0700440 flags.IncludeDirs = pathtools.PrefixPaths(c.properties.Include_dirs, ctx.Config().(Config).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,
524 deps CCDeps, subdir string, srcFiles []string) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800525
526 srcFiles = pathtools.PrefixPaths(srcFiles, common.ModuleSrcDir(ctx))
527 srcFiles = common.ExpandGlobs(ctx, srcFiles)
528
529 return TransformSourceToObj(ctx, subdir, srcFiles, ccFlagsToBuilderFlags(flags))
530}
531
532// Compile files listed in c.properties.Srcs into objects
Colin Cross97ba0732015-03-23 17:50:24 -0700533func (c *ccBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags,
534 deps CCDeps) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800535
536 if c.properties.SkipCompileObjs {
537 return nil
538 }
539
540 return c.customCompileObjs(ctx, flags, deps, "", c.properties.Srcs)
541}
542
Colin Cross5049f022015-03-18 13:28:46 -0700543// Compile generated source files from dependencies
Colin Cross97ba0732015-03-23 17:50:24 -0700544func (c *ccBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags,
545 deps CCDeps) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700546 var srcs []string
547
548 if c.properties.SkipCompileObjs {
549 return nil
550 }
551
552 ctx.VisitDirectDeps(func(module blueprint.Module) {
553 if gen, ok := module.(genrule.SourceFileGenerator); ok {
554 srcs = append(srcs, gen.GeneratedSourceFiles()...)
555 }
556 })
557
558 if len(srcs) == 0 {
559 return nil
560 }
561
562 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags))
563}
564
Colin Cross3f40fa42015-01-30 17:27:36 -0800565func (c *ccBase) outputFile() string {
566 return ""
567}
568
Colin Cross21b9a242015-03-24 14:15:58 -0700569func (c *ccBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800570 names []string) (modules []common.AndroidModule,
571 outputFiles []string, exportedIncludeDirs []string) {
572
573 for _, n := range names {
574 found := false
575 ctx.VisitDirectDeps(func(m blueprint.Module) {
576 otherName := ctx.OtherModuleName(m)
577 if otherName != n {
578 return
579 }
580
Colin Cross97ba0732015-03-23 17:50:24 -0700581 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800582 if a.Disabled() {
583 // If a cc_library host+device module depends on a library that exists as both
584 // cc_library_shared and cc_library_host_shared, it will end up with two
585 // dependencies with the same name, one of which is marked disabled for each
586 // of host and device. Ignore the disabled one.
587 return
588 }
589 if a.HostOrDevice() != ctx.Arch().HostOrDevice {
590 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
591 otherName)
592 return
593 }
594
595 if outputFile := a.outputFile(); outputFile != "" {
596 if found {
597 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
598 return
599 }
600 outputFiles = append(outputFiles, outputFile)
601 modules = append(modules, a)
602 if i, ok := a.(ccExportedIncludeDirsProducer); ok {
603 exportedIncludeDirs = append(exportedIncludeDirs, i.exportedIncludeDirs()...)
604 }
605 found = true
606 } else {
607 ctx.ModuleErrorf("module %q missing output file", otherName)
608 return
609 }
610 } else {
611 ctx.ModuleErrorf("module %q not an android module", otherName)
612 return
613 }
614 })
615 if !found {
616 ctx.ModuleErrorf("unsatisified dependency on %q", n)
617 }
618 }
619
620 return modules, outputFiles, exportedIncludeDirs
621}
622
Colin Cross21b9a242015-03-24 14:15:58 -0700623// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
624// containing paths
625func (c *ccBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
626 var depPaths CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800627 var newIncludeDirs []string
628
Colin Cross21b9a242015-03-24 14:15:58 -0700629 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800630
Colin Cross21b9a242015-03-24 14:15:58 -0700631 wholeStaticLibModules, depPaths.WholeStaticLibs, newIncludeDirs =
632 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
633 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800634
Colin Cross21b9a242015-03-24 14:15:58 -0700635 for _, m := range wholeStaticLibModules {
636 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
637 depPaths.WholeStaticLibObjFiles =
638 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
639 } else {
640 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
641 }
642 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800643
Colin Cross21b9a242015-03-24 14:15:58 -0700644 _, depPaths.StaticLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.StaticLibs)
645 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
646
647 _, depPaths.LateStaticLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
648 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
649
650 _, depPaths.SharedLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.SharedLibs)
651 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
652
653 ctx.VisitDirectDeps(func(m blueprint.Module) {
654 if obj, ok := m.(*ccObject); ok {
655 otherName := ctx.OtherModuleName(m)
656 if otherName == depNames.CrtBegin {
657 if !c.properties.Nocrt {
658 depPaths.CrtBegin = obj.outputFile()
659 }
660 } else if otherName == depNames.CrtEnd {
661 if !c.properties.Nocrt {
662 depPaths.CrtEnd = obj.outputFile()
663 }
664 } else {
665 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.outputFile())
666 }
667 }
668 })
669
670 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800671}
672
Colin Crossed4cf0b2015-03-26 14:43:45 -0700673// ccLinked contains the properties and members used by libraries and executables
674type ccLinked struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800675 ccBase
Colin Crossed4cf0b2015-03-26 14:43:45 -0700676
677 dynamicProperties struct {
678 VariantIsShared bool `blueprint:"mutated"`
679 VariantIsStatic bool `blueprint:"mutated"`
680 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800681}
682
Colin Crossed4cf0b2015-03-26 14:43:45 -0700683func newCCDynamic(dynamic *ccLinked, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700684 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
685
Colin Crossed4cf0b2015-03-26 14:43:45 -0700686 props = append(props, &dynamic.dynamicProperties)
687
Colin Crossc472d572015-03-17 15:06:21 -0700688 return newCCBase(&dynamic.ccBase, module, hod, multilib, props...)
689}
690
Colin Crossed4cf0b2015-03-26 14:43:45 -0700691func (c *ccLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross28d76592015-03-26 16:14:04 -0700692 if ctx.ContainsProperty("system_shared_libs") {
693 return c.properties.System_shared_libs
694 } else {
Colin Crossf6566ed2015-03-24 11:13:38 -0700695 if ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800696 return []string{}
Dan Albertbe961682015-03-18 23:38:50 -0700697 } else if c.properties.Sdk_version != "" {
698 version := c.properties.Sdk_version
699 libs := []string{
700 "ndk_libc." + version,
701 "ndk_libm." + version,
702 }
703
704 if c.properties.Sdk_version != "" && c.stl(ctx) == "ndk_system" {
705 libs = append([]string{"libstdc++"}, libs...)
706 }
707 return libs
Colin Cross3f40fa42015-01-30 17:27:36 -0800708 } else {
709 return []string{"libc", "libm"}
710 }
711 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800712}
713
Colin Crossed4cf0b2015-03-26 14:43:45 -0700714func (c *ccLinked) stl(ctx common.AndroidBaseContext) string {
715 if c.properties.Sdk_version != "" {
716 switch c.properties.Stl {
717 case "":
718 return "ndk_system"
719 case "c++_shared", "c++_static",
720 "stlport_shared", "stlport_static",
721 "gnustl_static":
722 return "ndk_lib" + c.properties.Stl
723 default:
724 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.properties.Stl)
725 return ""
726 }
727 }
728
729 switch c.properties.Stl {
730 case "libc++", "libc++_static",
731 "stlport", "stlport_static",
732 "libstdc++":
733 return c.properties.Stl
734 case "none":
735 return ""
736 case "":
737 if c.shared() {
738 return "libc++" // TODO: mingw needs libstdc++
739 } else {
740 return "libc++_static"
741 }
742 default:
743 ctx.ModuleErrorf("stl: %q is not a supported STL", c.properties.Stl)
744 return ""
745 }
746}
747
748func (c *ccLinked) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
749 stl := c.stl(ctx)
750 if ctx.Failed() {
751 return flags
752 }
753
754 switch stl {
755 case "libc++", "libc++_static":
756 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
757 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/external/libcxx/include")
758 if ctx.Host() {
759 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
760 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
761 flags.LdLibs = append(flags.LdLibs, "-lc", "-lm", "-lpthread")
762 }
763 case "stlport", "stlport_static":
764 if ctx.Device() {
765 flags.IncludeDirs = append(flags.IncludeDirs,
766 "${SrcDir}/external/stlport/stlport",
767 "${SrcDir}/bionic/libstdc++/include",
768 "${SrcDir}/bionic")
769 }
770 case "libstdc++":
771 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
772 // tree is in good enough shape to not need it.
773 // Host builds will use GNU libstdc++.
774 if ctx.Device() {
775 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/bionic/libstdc++/include")
776 }
777 case "ndk_system":
778 ndkSrcRoot := ctx.Config().(Config).SrcDir() + "/prebuilts/ndk/current/sources/"
779 flags.IncludeDirs = append(flags.IncludeDirs, ndkSrcRoot+"cxx-stl/system/include")
780 case "ndk_libc++_shared", "ndk_libc++_static":
781 // TODO(danalbert): This really shouldn't be here...
782 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
783 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
784 // Nothing
785 case "":
786 // None or error.
787 if ctx.Host() {
788 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
789 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
790 flags.LdLibs = append(flags.LdLibs, "-lc", "-lm")
791 }
792 default:
793 panic(fmt.Errorf("Unknown stl in ccLinked.Flags: %q", stl))
794 }
795
796 return flags
797}
798
799func (c *ccLinked) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700800 depNames = c.ccBase.DepNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800801
Colin Crossed4cf0b2015-03-26 14:43:45 -0700802 stl := c.stl(ctx)
803 if ctx.Failed() {
804 return depNames
805 }
806
807 switch stl {
808 case "libc++":
809 depNames.SharedLibs = append(depNames.SharedLibs, stl)
810 case "libstdc++":
811 if ctx.Device() {
812 depNames.SharedLibs = append(depNames.SharedLibs, stl)
813 }
814 case "libc++_static":
815 depNames.StaticLibs = append(depNames.StaticLibs, stl)
816 case "stlport":
817 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
818 case "stlport_static":
819 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
820 case "":
821 // None or error.
822 case "ndk_system":
823 // TODO: Make a system STL prebuilt for the NDK.
824 // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have
825 // its own includes. The includes are handled in ccBase.Flags().
826 case "ndk_libc++_shared", "ndk_libstlport_shared":
827 depNames.SharedLibs = append(depNames.SharedLibs, stl)
828 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
829 depNames.StaticLibs = append(depNames.StaticLibs, stl)
830 default:
831 panic(fmt.Errorf("Unknown stl in ccLinked.DepNames: %q", stl))
832 }
833
Colin Crossf6566ed2015-03-24 11:13:38 -0700834 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700835 if ctx.ModuleName() != "libcompiler_rt-extras" {
836 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
837 }
Colin Cross77b00fa2015-03-16 16:15:49 -0700838 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700839 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700840
841 if c.shared() {
842 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
843 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800844 }
845
Colin Cross21b9a242015-03-24 14:15:58 -0700846 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800847}
848
Colin Crossed4cf0b2015-03-26 14:43:45 -0700849// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
850type ccLinkedInterface interface {
851 // Returns true if the build options for the module have selected a static or shared build
852 buildStatic() bool
853 buildShared() bool
854
855 // Sets whether a specific variant is static or shared
856 setStatic()
857 setShared()
858
859 // Returns whether a specific variant is static or shared
860 static() bool
861 shared() bool
862}
863
864var _ ccLinkedInterface = (*CCLibrary)(nil)
865var _ ccLinkedInterface = (*CCBinary)(nil)
866
867func (c *ccLinked) static() bool {
868 return c.dynamicProperties.VariantIsStatic
869}
870
871func (c *ccLinked) shared() bool {
872 return c.dynamicProperties.VariantIsShared
873}
874
875func (c *ccLinked) setStatic() {
876 c.dynamicProperties.VariantIsStatic = true
877}
878
879func (c *ccLinked) setShared() {
880 c.dynamicProperties.VariantIsShared = true
881}
882
Colin Cross3f40fa42015-01-30 17:27:36 -0800883type ccExportedIncludeDirsProducer interface {
884 exportedIncludeDirs() []string
885}
886
887//
888// Combined static+shared libraries
889//
890
Colin Cross97ba0732015-03-23 17:50:24 -0700891type CCLibrary struct {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700892 ccLinked
Colin Cross3f40fa42015-01-30 17:27:36 -0800893
Colin Crossed4cf0b2015-03-26 14:43:45 -0700894 reuseFrom ccLibraryInterface
895 reuseObjFiles []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800896 objFiles []string
897 exportIncludeDirs []string
898 out string
899
Colin Cross97ba0732015-03-23 17:50:24 -0700900 LibraryProperties struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800901 BuildStatic bool `blueprint:"mutated"`
902 BuildShared bool `blueprint:"mutated"`
Colin Crossed4cf0b2015-03-26 14:43:45 -0700903 Static struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800904 Srcs []string `android:"arch_variant"`
905 Cflags []string `android:"arch_variant"`
906 } `android:"arch_variant"`
907 Shared struct {
908 Srcs []string `android:"arch_variant"`
909 Cflags []string `android:"arch_variant"`
910 } `android:"arch_variant"`
911 }
912}
913
Colin Crossed4cf0b2015-03-26 14:43:45 -0700914func (c *CCLibrary) buildStatic() bool {
915 return c.LibraryProperties.BuildStatic
916}
917
918func (c *CCLibrary) buildShared() bool {
919 return c.LibraryProperties.BuildShared
920}
921
Colin Cross97ba0732015-03-23 17:50:24 -0700922type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700923 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -0700924 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700925 setReuseFrom(ccLibraryInterface)
926 getReuseFrom() ccLibraryInterface
927 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -0700928 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -0700929}
930
Colin Crossed4cf0b2015-03-26 14:43:45 -0700931var _ ccLibraryInterface = (*CCLibrary)(nil)
932
Colin Cross97ba0732015-03-23 17:50:24 -0700933func (c *CCLibrary) ccLibrary() *CCLibrary {
934 return c
Colin Cross3f40fa42015-01-30 17:27:36 -0800935}
936
Colin Cross97ba0732015-03-23 17:50:24 -0700937func NewCCLibrary(library *CCLibrary, module CCModuleType,
938 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
939
Colin Crossed4cf0b2015-03-26 14:43:45 -0700940 return newCCDynamic(&library.ccLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -0700941 &library.LibraryProperties)
942}
943
944func CCLibraryFactory() (blueprint.Module, []interface{}) {
945 module := &CCLibrary{}
946
947 module.LibraryProperties.BuildShared = true
948 module.LibraryProperties.BuildStatic = true
949
950 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
951}
952
Colin Cross21b9a242015-03-24 14:15:58 -0700953func (c *CCLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700954 depNames = c.ccLinked.DepNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -0700955 if c.shared() {
Colin Crossf6566ed2015-03-24 11:13:38 -0700956 if ctx.Device() {
Colin Cross21b9a242015-03-24 14:15:58 -0700957 depNames.CrtBegin = "crtbegin_so"
958 depNames.CrtEnd = "crtend_so"
Colin Cross3f40fa42015-01-30 17:27:36 -0800959 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800960 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800961
Colin Cross21b9a242015-03-24 14:15:58 -0700962 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800963}
964
Colin Cross97ba0732015-03-23 17:50:24 -0700965func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800966 return c.out
967}
968
Colin Crossed4cf0b2015-03-26 14:43:45 -0700969func (c *CCLibrary) getReuseObjFiles() []string {
970 return c.reuseObjFiles
971}
972
973func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
974 c.reuseFrom = reuseFrom
975}
976
977func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
978 return c.reuseFrom
979}
980
Colin Cross97ba0732015-03-23 17:50:24 -0700981func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800982 return c.objFiles
983}
984
Colin Cross97ba0732015-03-23 17:50:24 -0700985func (c *CCLibrary) exportedIncludeDirs() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800986 return c.exportIncludeDirs
987}
988
Colin Cross21b9a242015-03-24 14:15:58 -0700989func (c *CCLibrary) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700990 flags = c.ccLinked.Flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -0700991
Colin Cross97ba0732015-03-23 17:50:24 -0700992 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -0800993
Colin Crossed4cf0b2015-03-26 14:43:45 -0700994 if c.shared() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800995 libName := ctx.ModuleName()
996 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
997 sharedFlag := "-Wl,-shared"
Colin Crossf6566ed2015-03-24 11:13:38 -0700998 if c.properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800999 sharedFlag = "-shared"
1000 }
Colin Crossf6566ed2015-03-24 11:13:38 -07001001 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001002 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001003 }
Colin Cross97ba0732015-03-23 17:50:24 -07001004
1005 flags.LdFlags = append(flags.LdFlags,
1006 "-Wl,--gc-sections",
1007 sharedFlag,
1008 "-Wl,-soname,"+libName+sharedLibraryExtension,
1009 )
Colin Cross3f40fa42015-01-30 17:27:36 -08001010 }
Colin Cross97ba0732015-03-23 17:50:24 -07001011
1012 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001013}
1014
Colin Cross97ba0732015-03-23 17:50:24 -07001015func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1016 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001017
1018 staticFlags := flags
Colin Cross97ba0732015-03-23 17:50:24 -07001019 staticFlags.CFlags = append(staticFlags.CFlags, c.LibraryProperties.Static.Cflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001020 objFilesStatic := c.customCompileObjs(ctx, staticFlags, deps, common.DeviceStaticLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001021 c.LibraryProperties.Static.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001022
1023 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001024 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001025
1026 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1027
1028 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1029
1030 c.objFiles = objFiles
1031 c.out = outputFile
1032 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1033 common.ModuleSrcDir(ctx))
1034
1035 ctx.CheckbuildFile(outputFile)
1036}
1037
Colin Cross97ba0732015-03-23 17:50:24 -07001038func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1039 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001040
1041 sharedFlags := flags
Colin Cross97ba0732015-03-23 17:50:24 -07001042 sharedFlags.CFlags = append(sharedFlags.CFlags, c.LibraryProperties.Shared.Cflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001043 objFilesShared := c.customCompileObjs(ctx, sharedFlags, deps, common.DeviceSharedLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001044 c.LibraryProperties.Shared.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001045
1046 objFiles = append(objFiles, objFilesShared...)
1047
1048 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1049
Colin Cross97ba0732015-03-23 17:50:24 -07001050 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001051 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, false,
Colin Cross77b00fa2015-03-16 16:15:49 -07001052 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001053
1054 c.out = outputFile
1055 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1056 common.ModuleSrcDir(ctx))
Colin Cross3f40fa42015-01-30 17:27:36 -08001057}
1058
Colin Cross97ba0732015-03-23 17:50:24 -07001059func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1060 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001061
1062 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001063 if c.getReuseFrom().ccLibrary() == c {
1064 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001065 } else {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001066 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001067 }
1068
Colin Crossed4cf0b2015-03-26 14:43:45 -07001069 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001070 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1071 } else {
1072 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1073 }
1074}
1075
Colin Cross97ba0732015-03-23 17:50:24 -07001076func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001077 // Static libraries do not get installed.
1078}
1079
Colin Cross97ba0732015-03-23 17:50:24 -07001080func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001081 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001082 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001083 installDir = "lib64"
1084 }
1085
1086 ctx.InstallFile(installDir, c.out)
1087}
1088
Colin Cross97ba0732015-03-23 17:50:24 -07001089func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001090 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001091 c.installStaticLibrary(ctx, flags)
1092 } else {
1093 c.installSharedLibrary(ctx, flags)
1094 }
1095}
1096
Colin Cross3f40fa42015-01-30 17:27:36 -08001097//
1098// Objects (for crt*.o)
1099//
1100
1101type ccObject struct {
1102 ccBase
1103 out string
1104}
1105
Colin Cross97ba0732015-03-23 17:50:24 -07001106func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001107 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001108
Colin Crossc472d572015-03-17 15:06:21 -07001109 return newCCBase(&module.ccBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001110}
1111
1112func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1113 // object files can't have any dynamic dependencies
1114 return nil
1115}
1116
Colin Cross21b9a242015-03-24 14:15:58 -07001117func (*ccObject) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1118 // object files can't have any dynamic dependencies
1119 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001120}
1121
1122func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001123 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001124
Colin Cross97ba0732015-03-23 17:50:24 -07001125 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001126
1127 var outputFile string
1128 if len(objFiles) == 1 {
1129 outputFile = objFiles[0]
1130 } else {
1131 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+".o")
1132 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1133 }
1134
1135 c.out = outputFile
1136
1137 ctx.CheckbuildFile(outputFile)
1138}
1139
Colin Cross97ba0732015-03-23 17:50:24 -07001140func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001141 // Object files do not get installed.
1142}
1143
Colin Cross3f40fa42015-01-30 17:27:36 -08001144func (c *ccObject) outputFile() string {
1145 return c.out
1146}
1147
1148//
1149// Executables
1150//
1151
Colin Cross97ba0732015-03-23 17:50:24 -07001152type CCBinary struct {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001153 ccLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001154 out string
Colin Cross97ba0732015-03-23 17:50:24 -07001155 BinaryProperties struct {
1156 // static_executable: compile executable with -static
1157 Static_executable bool
1158
1159 // stem: set the name of the output
1160 Stem string `android:"arch_variant"`
1161
Colin Cross4ae185c2015-03-26 15:12:10 -07001162 // suffix: append to the name of the output
1163 Suffix string `android:"arch_variant"`
1164
Colin Cross97ba0732015-03-23 17:50:24 -07001165 // prefix_symbols: if set, add an extra objcopy --prefix-symbols= step
1166 Prefix_symbols string
1167 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001168}
1169
Colin Crossed4cf0b2015-03-26 14:43:45 -07001170func (c *CCBinary) buildStatic() bool {
1171 return c.BinaryProperties.Static_executable
1172}
1173
1174func (c *CCBinary) buildShared() bool {
1175 return !c.BinaryProperties.Static_executable
1176}
1177
Colin Cross97ba0732015-03-23 17:50:24 -07001178func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001179 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001180 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001181 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001182 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001183
1184 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001185}
1186
Colin Cross21b9a242015-03-24 14:15:58 -07001187func (c *CCBinary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001188 depNames = c.ccLinked.DepNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001189 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001190 if c.BinaryProperties.Static_executable {
Colin Cross21b9a242015-03-24 14:15:58 -07001191 depNames.CrtBegin = "crtbegin_static"
Colin Cross3f40fa42015-01-30 17:27:36 -08001192 } else {
Colin Cross21b9a242015-03-24 14:15:58 -07001193 depNames.CrtBegin = "crtbegin_dynamic"
Colin Cross3f40fa42015-01-30 17:27:36 -08001194 }
Colin Cross21b9a242015-03-24 14:15:58 -07001195 depNames.CrtEnd = "crtend_android"
Colin Crossed4cf0b2015-03-26 14:43:45 -07001196
1197 if c.BinaryProperties.Static_executable {
1198 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1199 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1200 // move them to the beginning of deps.LateStaticLibs
1201 var groupLibs []string
1202 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1203 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1204 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1205 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001206 }
Colin Cross21b9a242015-03-24 14:15:58 -07001207 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001208}
1209
Colin Cross97ba0732015-03-23 17:50:24 -07001210func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001211 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001212
Colin Cross1f8f2342015-03-26 16:09:47 -07001213 props = append(props, &binary.BinaryProperties)
1214
1215 return newCCDynamic(&binary.ccLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001216}
1217
Colin Cross97ba0732015-03-23 17:50:24 -07001218func CCBinaryFactory() (blueprint.Module, []interface{}) {
1219 module := &CCBinary{}
1220
1221 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001222}
1223
Colin Cross21b9a242015-03-24 14:15:58 -07001224func (c *CCBinary) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001225 flags = c.ccLinked.Flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001226
Colin Cross97ba0732015-03-23 17:50:24 -07001227 flags.CFlags = append(flags.CFlags, "-fpie")
1228
Colin Crossf6566ed2015-03-24 11:13:38 -07001229 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001230 if c.BinaryProperties.Static_executable {
1231 // Clang driver needs -static to create static executable.
1232 // However, bionic/linker uses -shared to overwrite.
1233 // Linker for x86 targets does not allow coexistance of -static and -shared,
1234 // so we add -static only if -shared is not used.
1235 if !inList("-shared", flags.LdFlags) {
1236 flags.LdFlags = append(flags.LdFlags, "-static")
1237 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001238
Colin Crossed4cf0b2015-03-26 14:43:45 -07001239 flags.LdFlags = append(flags.LdFlags,
1240 "-nostdlib",
1241 "-Bstatic",
1242 "-Wl,--gc-sections",
1243 )
1244
1245 } else {
1246 linker := "/system/bin/linker"
1247 if flags.Toolchain.Is64Bit() {
1248 linker = "/system/bin/linker64"
1249 }
1250
1251 flags.LdFlags = append(flags.LdFlags,
1252 "-nostdlib",
1253 "-Bdynamic",
1254 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1255 "-Wl,--gc-sections",
1256 "-Wl,-z,nocopyreloc",
1257 )
1258 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001259 }
1260
Colin Cross97ba0732015-03-23 17:50:24 -07001261 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001262}
1263
Colin Cross97ba0732015-03-23 17:50:24 -07001264func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1265 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001266
Colin Cross97ba0732015-03-23 17:50:24 -07001267 if !c.BinaryProperties.Static_executable && inList("libc", c.properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001268 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1269 "from static libs or set static_executable: true")
1270 }
1271
1272 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001273 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001274 if c.BinaryProperties.Prefix_symbols != "" {
1275 afterPrefixSymbols := outputFile
1276 outputFile = outputFile + ".intermediate"
1277 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1278 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1279 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001280
Colin Cross97ba0732015-03-23 17:50:24 -07001281 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001282 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001283 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001284}
Colin Cross3f40fa42015-01-30 17:27:36 -08001285
Colin Cross97ba0732015-03-23 17:50:24 -07001286func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001287 ctx.InstallFile("bin", c.out)
1288}
1289
1290type ccTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001291 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001292
1293 testProperties struct {
1294 // test_per_src: Create a separate test for each source file. Useful when there is
1295 // global state that can not be torn down and reset between each test suite.
1296 Test_per_src bool
1297 }
Dan Albertc403f7c2015-03-18 14:01:18 -07001298}
1299
Colin Cross21b9a242015-03-24 14:15:58 -07001300func (c *ccTest) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
1301 flags = c.CCBinary.Flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001302
Colin Cross97ba0732015-03-23 17:50:24 -07001303 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001304 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001305 flags.CFlags = append(flags.CFlags, "-O0", "-g")
1306 flags.LdLibs = append(flags.LdLibs, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001307 }
1308
1309 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross97ba0732015-03-23 17:50:24 -07001310 flags.IncludeDirs = append(flags.IncludeDirs,
Tim Kilbourn5ccc7302015-03-19 10:02:21 -07001311 filepath.Join(ctx.Config().(Config).SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001312
Colin Cross21b9a242015-03-24 14:15:58 -07001313 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001314}
1315
Colin Cross21b9a242015-03-24 14:15:58 -07001316func (c *ccTest) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1317 depNames = c.CCBinary.DepNames(ctx, depNames)
1318 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
1319 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001320}
1321
Colin Cross97ba0732015-03-23 17:50:24 -07001322func (c *ccTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001323 if ctx.Device() {
Tim Kilbourn5ccc7302015-03-19 10:02:21 -07001324 ctx.InstallFile("../data/nativetest/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001325 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001326 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001327 }
1328}
1329
Colin Cross97ba0732015-03-23 17:50:24 -07001330func CCTestFactory() (blueprint.Module, []interface{}) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001331 module := &ccTest{}
Colin Cross1f8f2342015-03-26 16:09:47 -07001332 return NewCCBinary(&module.CCBinary, module, common.HostAndDeviceSupported,
1333 &module.testProperties)
Colin Cross6b290692015-03-19 14:05:33 -07001334}
1335
1336func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
1337 if test, ok := mctx.Module().(*ccTest); ok {
1338 if test.testProperties.Test_per_src {
1339 testNames := make([]string, len(test.properties.Srcs))
1340 for i, src := range test.properties.Srcs {
1341 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1342 }
1343 tests := mctx.CreateLocalVariations(testNames...)
1344 for i, src := range test.properties.Srcs {
1345 tests[i].(*ccTest).properties.Srcs = []string{src}
Colin Cross97ba0732015-03-23 17:50:24 -07001346 tests[i].(*ccTest).BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001347 }
1348 }
1349 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001350}
1351
1352//
1353// Static library
1354//
1355
Colin Cross97ba0732015-03-23 17:50:24 -07001356func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1357 module := &CCLibrary{}
1358 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001359
Colin Cross97ba0732015-03-23 17:50:24 -07001360 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001361}
1362
1363//
1364// Shared libraries
1365//
1366
Colin Cross97ba0732015-03-23 17:50:24 -07001367func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1368 module := &CCLibrary{}
1369 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001370
Colin Cross97ba0732015-03-23 17:50:24 -07001371 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001372}
1373
1374//
1375// Host static library
1376//
1377
Colin Cross97ba0732015-03-23 17:50:24 -07001378func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1379 module := &CCLibrary{}
1380 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001381
Colin Cross97ba0732015-03-23 17:50:24 -07001382 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001383}
1384
1385//
1386// Host Shared libraries
1387//
1388
Colin Cross97ba0732015-03-23 17:50:24 -07001389func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1390 module := &CCLibrary{}
1391 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001392
Colin Cross97ba0732015-03-23 17:50:24 -07001393 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001394}
1395
1396//
1397// Host Binaries
1398//
1399
Colin Cross97ba0732015-03-23 17:50:24 -07001400func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1401 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001402
Colin Cross97ba0732015-03-23 17:50:24 -07001403 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001404}
1405
1406//
Colin Cross1f8f2342015-03-26 16:09:47 -07001407// Host Tests
1408//
1409
1410func CCTestHostFactory() (blueprint.Module, []interface{}) {
1411 module := &ccTest{}
1412 return NewCCBinary(&module.CCBinary, module, common.HostSupported,
1413 &module.testProperties)
1414}
1415
1416//
Colin Cross3f40fa42015-01-30 17:27:36 -08001417// Device libraries shipped with gcc
1418//
1419
1420type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001421 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001422}
1423
1424func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1425 // toolchain libraries can't have any dependencies
1426 return nil
1427}
1428
Colin Cross21b9a242015-03-24 14:15:58 -07001429func (*toolchainLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001430 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001431 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001432}
1433
Colin Cross97ba0732015-03-23 17:50:24 -07001434func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001435 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001436
Colin Cross97ba0732015-03-23 17:50:24 -07001437 module.LibraryProperties.BuildStatic = true
1438
Colin Cross21b9a242015-03-24 14:15:58 -07001439 return newCCBase(&module.ccBase, module, common.DeviceSupported, common.MultilibBoth,
1440 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001441}
1442
1443func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001444 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001445
1446 libName := ctx.ModuleName() + staticLibraryExtension
1447 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1448
1449 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1450
1451 c.out = outputFile
1452
1453 ctx.CheckbuildFile(outputFile)
1454}
1455
Colin Cross97ba0732015-03-23 17:50:24 -07001456func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001457 // Toolchain libraries do not get installed.
1458}
1459
Dan Albertbe961682015-03-18 23:38:50 -07001460// NDK prebuilt libraries.
1461//
1462// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1463// either (with the exception of the shared STLs, which are installed to the app's directory rather
1464// than to the system image).
1465
1466func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1467 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
1468 ctx.Config().(Config).SrcDir(), version, toolchain.Name())
1469}
1470
1471type ndkPrebuiltLibrary struct {
1472 CCLibrary
1473}
1474
1475func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1476 ctx common.AndroidDynamicDependerModuleContext) []string {
1477
1478 // NDK libraries can't have any dependencies
1479 return nil
1480}
1481
1482func (*ndkPrebuiltLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1483 // NDK libraries can't have any dependencies
1484 return CCDeps{}
1485}
1486
1487func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1488 module := &ndkPrebuiltLibrary{}
1489 module.LibraryProperties.BuildShared = true
1490 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1491}
1492
1493func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1494 deps CCDeps, objFiles []string) {
1495 // A null build step, but it sets up the output path.
1496 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1497 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1498 }
1499
1500 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1501 common.ModuleSrcDir(ctx))
1502
1503 // NDK prebuilt libraries are named like: ndk_LIBNAME.SDK_VERSION.
1504 // We want to translate to just LIBNAME.
1505 libName := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1506 libDir := getNdkLibDir(ctx, flags.Toolchain, c.properties.Sdk_version)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001507 c.out = filepath.Join(libDir, libName+sharedLibraryExtension)
Dan Albertbe961682015-03-18 23:38:50 -07001508}
1509
1510func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1511 // Toolchain libraries do not get installed.
1512}
1513
1514// The NDK STLs are slightly different from the prebuilt system libraries:
1515// * Are not specific to each platform version.
1516// * The libraries are not in a predictable location for each STL.
1517
1518type ndkPrebuiltStl struct {
1519 ndkPrebuiltLibrary
1520}
1521
1522type ndkPrebuiltStaticStl struct {
1523 ndkPrebuiltStl
1524}
1525
1526type ndkPrebuiltSharedStl struct {
1527 ndkPrebuiltStl
1528}
1529
1530func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1531 module := &ndkPrebuiltSharedStl{}
1532 module.LibraryProperties.BuildShared = true
1533 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1534}
1535
1536func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1537 module := &ndkPrebuiltStaticStl{}
1538 module.LibraryProperties.BuildStatic = true
1539 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1540}
1541
1542func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1543 gccVersion := toolchain.GccVersion()
1544 var libDir string
1545 switch stl {
1546 case "libstlport":
1547 libDir = "cxx-stl/stlport/libs"
1548 case "libc++":
1549 libDir = "cxx-stl/llvm-libc++/libs"
1550 case "libgnustl":
1551 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1552 }
1553
1554 if libDir != "" {
1555 ndkSrcRoot := ctx.Config().(Config).SrcDir() + "/prebuilts/ndk/current/sources"
1556 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1557 }
1558
1559 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1560 return ""
1561}
1562
1563func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1564 deps CCDeps, objFiles []string) {
1565 // A null build step, but it sets up the output path.
1566 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1567 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1568 }
1569
1570 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1571 common.ModuleSrcDir(ctx))
1572
1573 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1574 libExt := sharedLibraryExtension
1575 if c.LibraryProperties.BuildStatic {
1576 libExt = staticLibraryExtension
1577 }
1578
1579 stlName := strings.TrimSuffix(libName, "_shared")
1580 stlName = strings.TrimSuffix(stlName, "_static")
1581 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1582 c.out = libDir + "/" + libName + libExt
1583}
1584
Colin Cross3f40fa42015-01-30 17:27:36 -08001585func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001586 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001587 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001588 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001589 modules = mctx.CreateLocalVariations("static", "shared")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001590 modules[0].(ccLinkedInterface).setStatic()
1591 modules[1].(ccLinkedInterface).setShared()
1592 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001593 modules = mctx.CreateLocalVariations("static")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001594 modules[0].(ccLinkedInterface).setStatic()
1595 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001596 modules = mctx.CreateLocalVariations("shared")
Colin Crossed4cf0b2015-03-26 14:43:45 -07001597 modules[0].(ccLinkedInterface).setShared()
Colin Cross3f40fa42015-01-30 17:27:36 -08001598 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001599 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001600 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001601
1602 if _, ok := c.(ccLibraryInterface); ok {
1603 reuseFrom := modules[0].(ccLibraryInterface)
1604 for _, m := range modules {
1605 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001606 }
1607 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001608 }
1609}