blob: 61e256847fc57fc192f100e13233bc2d8870ece0 [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"
Colin Cross0af4b842015-04-30 16:36:18 -070024 "runtime"
Colin Cross3f40fa42015-01-30 17:27:36 -080025 "strings"
26
Colin Cross97ba0732015-03-23 17:50:24 -070027 "github.com/google/blueprint"
28 "github.com/google/blueprint/pathtools"
29
Colin Cross3f40fa42015-01-30 17:27:36 -080030 "android/soong/common"
Colin Cross5049f022015-03-18 13:28:46 -070031 "android/soong/genrule"
Colin Cross3f40fa42015-01-30 17:27:36 -080032)
33
Colin Cross3f40fa42015-01-30 17:27:36 -080034var (
Colin Cross1332b002015-04-07 17:11:30 -070035 HostPrebuiltTag = pctx.VariableConfigMethod("HostPrebuiltTag", common.Config.PrebuiltOS)
36 SrcDir = pctx.VariableConfigMethod("SrcDir", common.Config.SrcDir)
Colin Cross3f40fa42015-01-30 17:27:36 -080037
38 LibcRoot = pctx.StaticVariable("LibcRoot", "${SrcDir}/bionic/libc")
39 LibmRoot = pctx.StaticVariable("LibmRoot", "${SrcDir}/bionic/libm")
40)
41
42// Flags used by lots of devices. Putting them in package static variables will save bytes in
43// build.ninja so they aren't repeated for every file
44var (
45 commonGlobalCflags = []string{
46 "-DANDROID",
47 "-fmessage-length=0",
48 "-W",
49 "-Wall",
50 "-Wno-unused",
51 "-Winit-self",
52 "-Wpointer-arith",
53
54 // COMMON_RELEASE_CFLAGS
55 "-DNDEBUG",
56 "-UDEBUG",
57 }
58
59 deviceGlobalCflags = []string{
60 // TARGET_ERROR_FLAGS
61 "-Werror=return-type",
62 "-Werror=non-virtual-dtor",
63 "-Werror=address",
64 "-Werror=sequence-point",
65 }
66
67 hostGlobalCflags = []string{}
68
69 commonGlobalCppflags = []string{
70 "-Wsign-promo",
71 "-std=gnu++11",
72 }
73)
74
75func init() {
76 pctx.StaticVariable("commonGlobalCflags", strings.Join(commonGlobalCflags, " "))
77 pctx.StaticVariable("deviceGlobalCflags", strings.Join(deviceGlobalCflags, " "))
78 pctx.StaticVariable("hostGlobalCflags", strings.Join(hostGlobalCflags, " "))
79
80 pctx.StaticVariable("commonGlobalCppflags", strings.Join(commonGlobalCppflags, " "))
81
82 pctx.StaticVariable("commonClangGlobalCflags",
83 strings.Join(clangFilterUnknownCflags(commonGlobalCflags), " "))
84 pctx.StaticVariable("deviceClangGlobalCflags",
85 strings.Join(clangFilterUnknownCflags(deviceGlobalCflags), " "))
86 pctx.StaticVariable("hostClangGlobalCflags",
87 strings.Join(clangFilterUnknownCflags(hostGlobalCflags), " "))
Tim Kilbournf2948142015-03-11 12:03:03 -070088 pctx.StaticVariable("commonClangGlobalCppflags",
89 strings.Join(clangFilterUnknownCflags(commonGlobalCppflags), " "))
Colin Cross3f40fa42015-01-30 17:27:36 -080090
91 // Everything in this list is a crime against abstraction and dependency tracking.
92 // Do not add anything to this list.
93 pctx.StaticVariable("commonGlobalIncludes", strings.Join([]string{
94 "-isystem ${SrcDir}/system/core/include",
95 "-isystem ${SrcDir}/hardware/libhardware/include",
96 "-isystem ${SrcDir}/hardware/libhardware_legacy/include",
97 "-isystem ${SrcDir}/hardware/ril/include",
98 "-isystem ${SrcDir}/libnativehelper/include",
99 "-isystem ${SrcDir}/frameworks/native/include",
100 "-isystem ${SrcDir}/frameworks/native/opengl/include",
101 "-isystem ${SrcDir}/frameworks/av/include",
102 "-isystem ${SrcDir}/frameworks/base/include",
103 }, " "))
104
105 pctx.StaticVariable("clangPath", "${SrcDir}/prebuilts/clang/${HostPrebuiltTag}/host/3.6/bin/")
106}
107
Colin Cross3f40fa42015-01-30 17:27:36 -0800108// Building C/C++ code is handled by objects that satisfy this interface via composition
Colin Cross97ba0732015-03-23 17:50:24 -0700109type CCModuleType interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800110 common.AndroidModule
111
Colin Crossfa138792015-04-24 17:31:52 -0700112 // Modify property values after parsing Blueprints file but before starting dependency
113 // resolution or build rule generation
114 ModifyProperties(common.AndroidBaseContext)
115
Colin Cross21b9a242015-03-24 14:15:58 -0700116 // Modify the ccFlags
Colin Cross0676e2d2015-04-24 17:39:18 -0700117 flags(common.AndroidModuleContext, CCFlags) CCFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800118
Colin Cross21b9a242015-03-24 14:15:58 -0700119 // Return list of dependency names for use in AndroidDynamicDependencies and in depsToPaths
Colin Cross0676e2d2015-04-24 17:39:18 -0700120 depNames(common.AndroidBaseContext, CCDeps) CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800121
122 // Compile objects into final module
Colin Cross97ba0732015-03-23 17:50:24 -0700123 compileModule(common.AndroidModuleContext, CCFlags, CCDeps, []string)
Colin Cross3f40fa42015-01-30 17:27:36 -0800124
Dan Albertc403f7c2015-03-18 14:01:18 -0700125 // Install the built module.
Colin Cross97ba0732015-03-23 17:50:24 -0700126 installModule(common.AndroidModuleContext, CCFlags)
Dan Albertc403f7c2015-03-18 14:01:18 -0700127
Colin Cross3f40fa42015-01-30 17:27:36 -0800128 // Return the output file (.o, .a or .so) for use by other modules
129 outputFile() string
130}
131
Colin Cross97ba0732015-03-23 17:50:24 -0700132type CCDeps struct {
Colin Cross28344522015-04-22 13:07:53 -0700133 StaticLibs, SharedLibs, LateStaticLibs, WholeStaticLibs, ObjFiles, Cflags []string
Colin Crossc472d572015-03-17 15:06:21 -0700134
Colin Cross21b9a242015-03-24 14:15:58 -0700135 WholeStaticLibObjFiles []string
136
Colin Cross97ba0732015-03-23 17:50:24 -0700137 CrtBegin, CrtEnd string
Colin Crossc472d572015-03-17 15:06:21 -0700138}
139
Colin Cross97ba0732015-03-23 17:50:24 -0700140type CCFlags struct {
Colin Cross28344522015-04-22 13:07:53 -0700141 GlobalFlags []string // Flags that apply to C, C++, and assembly source files
142 AsFlags []string // Flags that apply to assembly source files
143 CFlags []string // Flags that apply to C and C++ source files
144 ConlyFlags []string // Flags that apply to C source files
145 CppFlags []string // Flags that apply to C++ source files
146 YaccFlags []string // Flags that apply to Yacc source files
147 LdFlags []string // Flags that apply to linker command lines
148
149 Nocrt bool
150 Toolchain Toolchain
151 Clang bool
Colin Crossc472d572015-03-17 15:06:21 -0700152}
153
Colin Crossfa138792015-04-24 17:31:52 -0700154// CCBase contains the properties and members used by all C/C++ module types, and implements
Colin Crossc472d572015-03-17 15:06:21 -0700155// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
156// and uses a ccModuleType interface to that struct to create the build steps.
Colin Crossfa138792015-04-24 17:31:52 -0700157type CCBase struct {
Colin Crossc472d572015-03-17 15:06:21 -0700158 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700159 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700160
Colin Crossfa138792015-04-24 17:31:52 -0700161 // Properties used to compile all C or C++ modules
162 Properties struct {
163 // srcs: list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
164 Srcs []string `android:"arch_variant,arch_subtract"`
165
166 // cflags: list of module-specific flags that will be used for C and C++ compiles.
167 Cflags []string `android:"arch_variant"`
168
169 // cppflags: list of module-specific flags that will be used for C++ compiles
170 Cppflags []string `android:"arch_variant"`
171
172 // conlyflags: list of module-specific flags that will be used for C compiles
173 Conlyflags []string `android:"arch_variant"`
174
175 // asflags: list of module-specific flags that will be used for .S compiles
176 Asflags []string `android:"arch_variant"`
177
178 // yaccflags: list of module-specific flags that will be used for .y and .yy compiles
179 Yaccflags []string
180
181 // ldflags: list of module-specific flags that will be used for all link steps
182 Ldflags []string `android:"arch_variant"`
183
184 // instruction_set: the instruction set architecture to use to compile the C/C++
185 // module.
186 Instruction_set string `android:"arch_variant"`
187
188 // include_dirs: list of directories relative to the root of the source tree that will
189 // be added to the include path using -I.
190 // If possible, don't use this. If adding paths from the current directory use
191 // local_include_dirs, if adding paths from other modules use export_include_dirs in
192 // that module.
193 Include_dirs []string `android:"arch_variant"`
194
195 // local_include_dirs: list of directories relative to the Blueprints file that will
196 // be added to the include path using -I
197 Local_include_dirs []string `android:"arch_variant"`
198
199 // export_include_dirs: list of directories relative to the Blueprints file that will
200 // be added to the include path using -I for any module that links against this module
201 Export_include_dirs []string `android:"arch_variant"`
202
203 // clang_cflags: list of module-specific flags that will be used for C and C++ compiles when
204 // compiling with clang
205 Clang_cflags []string `android:"arch_variant"`
206
207 // clang_asflags: list of module-specific flags that will be used for .S compiles when
208 // compiling with clang
209 Clang_asflags []string `android:"arch_variant"`
210
211 // system_shared_libs: list of system libraries that will be dynamically linked to
212 // shared library and executable modules. If unset, generally defaults to libc
213 // and libm. Set to [] to prevent linking against libc and libm.
214 System_shared_libs []string
215
216 // whole_static_libs: list of modules whose object files should be linked into this module
217 // in their entirety. For static library modules, all of the .o files from the intermediate
218 // directory of the dependency will be linked into this modules .a file. For a shared library,
219 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
220 Whole_static_libs []string `android:"arch_variant"`
221
222 // static_libs: list of modules that should be statically linked into this module.
223 Static_libs []string `android:"arch_variant"`
224
225 // shared_libs: list of modules that should be dynamically linked into this module.
226 Shared_libs []string `android:"arch_variant"`
227
228 // allow_undefined_symbols: allow the module to contain undefined symbols. By default,
229 // modules cannot contain undefined symbols that are not satisified by their immediate
230 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
231 // This flag should only be necessary for compiling low-level libraries like libc.
232 Allow_undefined_symbols bool
233
234 // nocrt: don't link in crt_begin and crt_end. This flag should only be necessary for
235 // compiling crt or libc.
236 Nocrt bool `android:"arch_variant"`
237
238 // no_default_compiler_flags: don't insert default compiler flags into asflags, cflags,
239 // cppflags, conlyflags, ldflags, or include_dirs
240 No_default_compiler_flags bool
241
242 // clang: compile module with clang instead of gcc
243 Clang bool `android:"arch_variant"`
244
245 // rtti: pass -frtti instead of -fno-rtti
246 Rtti bool
247
248 // host_ldlibs: -l arguments to pass to linker for host-provided shared libraries
249 Host_ldlibs []string `android:"arch_variant"`
250
251 // stl: select the STL library to use. Possible values are "libc++", "libc++_static",
252 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
253 // default
254 Stl string
255
256 // Set for combined shared/static libraries to prevent compiling object files a second time
257 SkipCompileObjs bool `blueprint:"mutated"`
258
259 Debug struct {
260 Cflags []string `android:"arch_variant"`
261 } `android:"arch_variant"`
262 Release struct {
263 Cflags []string `android:"arch_variant"`
264 } `android:"arch_variant"`
265
266 // Minimum sdk version supported when compiling against the ndk
267 Sdk_version string
268
269 // relative_install_path: install to a subdirectory of the default install path for the module
270 Relative_install_path string
271 }
272
273 unused struct {
274 Asan bool
275 Native_coverage bool
276 Strip string
277 Tags []string
278 Required []string
279 }
Colin Crossc472d572015-03-17 15:06:21 -0700280
281 installPath string
Colin Cross74d1ec02015-04-28 13:30:13 -0700282
283 savedDepNames CCDeps
Colin Crossc472d572015-03-17 15:06:21 -0700284}
285
Colin Crossfa138792015-04-24 17:31:52 -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
Colin Crossfa138792015-04-24 17:31:52 -0700291 props = append(props, &base.Properties, &base.unused)
Colin Crossc472d572015-03-17 15:06:21 -0700292
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 Crossfa138792015-04-24 17:31:52 -0700296func (c *CCBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800297 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 Cross74d1ec02015-04-28 13:30:13 -0700307 deps := c.depsToPaths(ctx, c.savedDepNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800308 if ctx.Failed() {
309 return
310 }
311
Colin Cross28344522015-04-22 13:07:53 -0700312 flags.CFlags = append(flags.CFlags, deps.Cflags...)
Colin Crossed9f8682015-03-18 17:17:35 -0700313
Colin Cross581c1892015-04-07 16:50:10 -0700314 objFiles := c.compileObjs(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800315 if ctx.Failed() {
316 return
317 }
318
Colin Cross581c1892015-04-07 16:50:10 -0700319 generatedObjFiles := c.compileGeneratedObjs(ctx, flags)
Colin Cross5049f022015-03-18 13:28:46 -0700320 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 Crossfa138792015-04-24 17:31:52 -0700337func (c *CCBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800338 return c.module
339}
340
Colin Crossfa138792015-04-24 17:31:52 -0700341var _ common.AndroidDynamicDepender = (*CCBase)(nil)
Colin Cross3f40fa42015-01-30 17:27:36 -0800342
Colin Crossfa138792015-04-24 17:31:52 -0700343func (c *CCBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800344 arch := ctx.Arch()
Colin Crossd3ba0392015-05-07 14:11:29 -0700345 hod := ctx.HostOrDevice()
346 factory := toolchainFactories[hod][arch.ArchType]
Colin Cross3f40fa42015-01-30 17:27:36 -0800347 if factory == nil {
348 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
Colin Crossd3ba0392015-05-07 14:11:29 -0700349 hod.String(), arch.String()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800350 }
351 return factory(arch.ArchVariant, arch.CpuVariant)
352}
353
Colin Crossfa138792015-04-24 17:31:52 -0700354func (c *CCBase) ModifyProperties(ctx common.AndroidBaseContext) {
355}
356
Colin Crosse11befc2015-04-27 17:49:17 -0700357func (c *CCBase) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossfa138792015-04-24 17:31:52 -0700358 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.Properties.Whole_static_libs...)
359 depNames.StaticLibs = append(depNames.StaticLibs, c.Properties.Static_libs...)
360 depNames.SharedLibs = append(depNames.SharedLibs, c.Properties.Shared_libs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700361
Colin Cross21b9a242015-03-24 14:15:58 -0700362 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800363}
364
Colin Crossfa138792015-04-24 17:31:52 -0700365func (c *CCBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
366 c.module.ModifyProperties(ctx)
367
Colin Cross74d1ec02015-04-28 13:30:13 -0700368 c.savedDepNames = c.module.depNames(ctx, CCDeps{})
369 c.savedDepNames.WholeStaticLibs = lastUniqueElements(c.savedDepNames.WholeStaticLibs)
370 c.savedDepNames.StaticLibs = lastUniqueElements(c.savedDepNames.StaticLibs)
371 c.savedDepNames.SharedLibs = lastUniqueElements(c.savedDepNames.SharedLibs)
372
373 staticLibs := c.savedDepNames.WholeStaticLibs
374 staticLibs = append(staticLibs, c.savedDepNames.StaticLibs...)
375 staticLibs = append(staticLibs, c.savedDepNames.LateStaticLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700376 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800377
Colin Cross74d1ec02015-04-28 13:30:13 -0700378 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, c.savedDepNames.SharedLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700379
Colin Cross74d1ec02015-04-28 13:30:13 -0700380 ret := append([]string(nil), c.savedDepNames.ObjFiles...)
381 if c.savedDepNames.CrtBegin != "" {
382 ret = append(ret, c.savedDepNames.CrtBegin)
Colin Cross21b9a242015-03-24 14:15:58 -0700383 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700384 if c.savedDepNames.CrtEnd != "" {
385 ret = append(ret, c.savedDepNames.CrtEnd)
Colin Cross21b9a242015-03-24 14:15:58 -0700386 }
387
388 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800389}
390
391// Create a ccFlags struct that collects the compile flags from global values,
392// per-target values, module type values, and per-module Blueprints properties
Colin Crossfa138792015-04-24 17:31:52 -0700393func (c *CCBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700394 flags := CCFlags{
Colin Crossfa138792015-04-24 17:31:52 -0700395 CFlags: c.Properties.Cflags,
396 CppFlags: c.Properties.Cppflags,
397 ConlyFlags: c.Properties.Conlyflags,
398 LdFlags: c.Properties.Ldflags,
399 AsFlags: c.Properties.Asflags,
400 YaccFlags: c.Properties.Yaccflags,
401 Nocrt: c.Properties.Nocrt,
Colin Cross97ba0732015-03-23 17:50:24 -0700402 Toolchain: toolchain,
Colin Crossfa138792015-04-24 17:31:52 -0700403 Clang: c.Properties.Clang,
Colin Cross3f40fa42015-01-30 17:27:36 -0800404 }
Colin Cross28344522015-04-22 13:07:53 -0700405
406 // Include dir cflags
Colin Crossf2298272015-05-12 11:36:53 -0700407 common.CheckSrcDirsExist(ctx, c.Properties.Include_dirs, "include_dirs")
408 common.CheckModuleSrcDirsExist(ctx, c.Properties.Local_include_dirs, "local_include_dirs")
409
Colin Crossfa138792015-04-24 17:31:52 -0700410 rootIncludeDirs := pathtools.PrefixPaths(c.Properties.Include_dirs, ctx.AConfig().SrcDir())
411 localIncludeDirs := pathtools.PrefixPaths(c.Properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -0700412 flags.GlobalFlags = append(flags.GlobalFlags,
413 includeDirsToFlags(rootIncludeDirs),
414 includeDirsToFlags(localIncludeDirs))
415
Colin Crossfa138792015-04-24 17:31:52 -0700416 if !c.Properties.No_default_compiler_flags {
417 if c.Properties.Sdk_version == "" || ctx.Host() {
Colin Cross28344522015-04-22 13:07:53 -0700418 flags.GlobalFlags = append(flags.GlobalFlags,
419 "${commonGlobalIncludes}",
420 toolchain.IncludeFlags(),
421 "-I${SrcDir}/libnativehelper/include/nativehelper")
422 }
423
424 flags.GlobalFlags = append(flags.GlobalFlags, []string{
425 "-I" + common.ModuleSrcDir(ctx),
426 "-I" + common.ModuleOutDir(ctx),
427 "-I" + common.ModuleGenDir(ctx),
428 }...)
429 }
430
Colin Crossfa138792015-04-24 17:31:52 -0700431 instructionSet := c.Properties.Instruction_set
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700432 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
433 if err != nil {
434 ctx.ModuleErrorf("%s", err)
435 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800436
Colin Crossaf19a292015-03-18 12:07:10 -0700437 // TODO: debug
Colin Crossfa138792015-04-24 17:31:52 -0700438 flags.CFlags = append(flags.CFlags, c.Properties.Release.Cflags...)
Colin Crossaf19a292015-03-18 12:07:10 -0700439
Colin Cross28d76592015-03-26 16:14:04 -0700440 if ctx.Host() && !ctx.ContainsProperty("clang") {
Colin Cross97ba0732015-03-23 17:50:24 -0700441 flags.Clang = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800442 }
443
Colin Cross97ba0732015-03-23 17:50:24 -0700444 if flags.Clang {
445 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
Colin Crossfa138792015-04-24 17:31:52 -0700446 flags.CFlags = append(flags.CFlags, c.Properties.Clang_cflags...)
447 flags.AsFlags = append(flags.AsFlags, c.Properties.Clang_asflags...)
Colin Cross97ba0732015-03-23 17:50:24 -0700448 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
449 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
450 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800451
Colin Cross97ba0732015-03-23 17:50:24 -0700452 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
453 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700454 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700455 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700456 }
457
Colin Cross3f40fa42015-01-30 17:27:36 -0800458 target := "-target " + toolchain.ClangTriple()
459 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
460
Colin Cross97ba0732015-03-23 17:50:24 -0700461 flags.CFlags = append(flags.CFlags, target, gccPrefix)
462 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
463 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800464 }
465
Colin Crossfa138792015-04-24 17:31:52 -0700466 if !c.Properties.No_default_compiler_flags {
467 if ctx.Device() && !c.Properties.Allow_undefined_symbols {
Colin Cross97ba0732015-03-23 17:50:24 -0700468 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800469 }
470
Colin Cross56b4d452015-04-21 17:38:44 -0700471 flags.GlobalFlags = append(flags.GlobalFlags, instructionSetFlags)
472
Colin Cross97ba0732015-03-23 17:50:24 -0700473 if flags.Clang {
474 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700475 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800476 toolchain.ClangCflags(),
477 "${commonClangGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700478 fmt.Sprintf("${%sClangGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800479 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700480 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700481 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800482 toolchain.Cflags(),
483 "${commonGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700484 fmt.Sprintf("${%sGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800485 }
486
Colin Crossf6566ed2015-03-24 11:13:38 -0700487 if ctx.Device() {
Colin Crossfa138792015-04-24 17:31:52 -0700488 if c.Properties.Rtti {
Colin Cross97ba0732015-03-23 17:50:24 -0700489 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800490 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700491 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800492 }
493 }
494
Colin Cross97ba0732015-03-23 17:50:24 -0700495 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800496
Colin Cross97ba0732015-03-23 17:50:24 -0700497 if flags.Clang {
498 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
499 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800500 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700501 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
502 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800503 }
Colin Cross28344522015-04-22 13:07:53 -0700504
505 if ctx.Host() {
Colin Crossfa138792015-04-24 17:31:52 -0700506 flags.LdFlags = append(flags.LdFlags, c.Properties.Host_ldlibs...)
Colin Cross28344522015-04-22 13:07:53 -0700507 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800508 }
509
Colin Cross0676e2d2015-04-24 17:39:18 -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 Cross0676e2d2015-04-24 17:39:18 -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 Crossfa138792015-04-24 17:31:52 -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
Colin Crossfce53272015-04-08 11:21:40 -0700534 srcFiles = common.ExpandSources(ctx, srcFiles)
Colin Cross581c1892015-04-07 16:50:10 -0700535 srcFiles, deps := genSources(ctx, srcFiles, buildFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800536
Colin Cross581c1892015-04-07 16:50:10 -0700537 return TransformSourceToObj(ctx, subdir, srcFiles, buildFlags, deps)
Colin Cross3f40fa42015-01-30 17:27:36 -0800538}
539
Colin Crossfa138792015-04-24 17:31:52 -0700540// Compile files listed in c.Properties.Srcs into objects
541func (c *CCBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800542
Colin Crossfa138792015-04-24 17:31:52 -0700543 if c.Properties.SkipCompileObjs {
Colin Cross3f40fa42015-01-30 17:27:36 -0800544 return nil
545 }
546
Colin Crossfa138792015-04-24 17:31:52 -0700547 return c.customCompileObjs(ctx, flags, "", c.Properties.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800548}
549
Colin Cross5049f022015-03-18 13:28:46 -0700550// Compile generated source files from dependencies
Colin Crossfa138792015-04-24 17:31:52 -0700551func (c *CCBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700552 var srcs []string
553
Colin Crossfa138792015-04-24 17:31:52 -0700554 if c.Properties.SkipCompileObjs {
Colin Cross5049f022015-03-18 13:28:46 -0700555 return nil
556 }
557
558 ctx.VisitDirectDeps(func(module blueprint.Module) {
559 if gen, ok := module.(genrule.SourceFileGenerator); ok {
560 srcs = append(srcs, gen.GeneratedSourceFiles()...)
561 }
562 })
563
564 if len(srcs) == 0 {
565 return nil
566 }
567
Colin Cross581c1892015-04-07 16:50:10 -0700568 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags), nil)
Colin Cross5049f022015-03-18 13:28:46 -0700569}
570
Colin Crossfa138792015-04-24 17:31:52 -0700571func (c *CCBase) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800572 return ""
573}
574
Colin Crossfa138792015-04-24 17:31:52 -0700575func (c *CCBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800576 names []string) (modules []common.AndroidModule,
Colin Cross28344522015-04-22 13:07:53 -0700577 outputFiles []string, exportedFlags []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800578
579 for _, n := range names {
580 found := false
581 ctx.VisitDirectDeps(func(m blueprint.Module) {
582 otherName := ctx.OtherModuleName(m)
583 if otherName != n {
584 return
585 }
586
Colin Cross97ba0732015-03-23 17:50:24 -0700587 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800588 if a.Disabled() {
589 // If a cc_library host+device module depends on a library that exists as both
590 // cc_library_shared and cc_library_host_shared, it will end up with two
591 // dependencies with the same name, one of which is marked disabled for each
592 // of host and device. Ignore the disabled one.
593 return
594 }
Colin Crossd3ba0392015-05-07 14:11:29 -0700595 if a.HostOrDevice() != ctx.HostOrDevice() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800596 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
597 otherName)
598 return
599 }
600
601 if outputFile := a.outputFile(); outputFile != "" {
602 if found {
603 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
604 return
605 }
606 outputFiles = append(outputFiles, outputFile)
607 modules = append(modules, a)
Colin Cross28344522015-04-22 13:07:53 -0700608 if i, ok := a.(ccExportedFlagsProducer); ok {
609 exportedFlags = append(exportedFlags, i.exportedFlags()...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800610 }
611 found = true
612 } else {
613 ctx.ModuleErrorf("module %q missing output file", otherName)
614 return
615 }
616 } else {
617 ctx.ModuleErrorf("module %q not an android module", otherName)
618 return
619 }
620 })
621 if !found {
622 ctx.ModuleErrorf("unsatisified dependency on %q", n)
623 }
624 }
625
Colin Cross28344522015-04-22 13:07:53 -0700626 return modules, outputFiles, exportedFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800627}
628
Colin Cross21b9a242015-03-24 14:15:58 -0700629// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
630// containing paths
Colin Crossfa138792015-04-24 17:31:52 -0700631func (c *CCBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700632 var depPaths CCDeps
Colin Cross28344522015-04-22 13:07:53 -0700633 var newCflags []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800634
Colin Cross21b9a242015-03-24 14:15:58 -0700635 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800636
Colin Cross28344522015-04-22 13:07:53 -0700637 wholeStaticLibModules, depPaths.WholeStaticLibs, newCflags =
Colin Cross21b9a242015-03-24 14:15:58 -0700638 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
Colin Cross28344522015-04-22 13:07:53 -0700639 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800640
Colin Cross21b9a242015-03-24 14:15:58 -0700641 for _, m := range wholeStaticLibModules {
642 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
643 depPaths.WholeStaticLibObjFiles =
644 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
645 } else {
646 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
647 }
648 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800649
Colin Cross28344522015-04-22 13:07:53 -0700650 _, depPaths.StaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.StaticLibs)
651 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700652
Colin Cross28344522015-04-22 13:07:53 -0700653 _, depPaths.LateStaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
654 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700655
Colin Cross28344522015-04-22 13:07:53 -0700656 _, depPaths.SharedLibs, newCflags = c.depsToPathsFromList(ctx, depNames.SharedLibs)
657 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700658
659 ctx.VisitDirectDeps(func(m blueprint.Module) {
Dan Albertc3144b12015-04-28 18:17:56 -0700660 if obj, ok := m.(ccObjectProvider); ok {
Colin Cross21b9a242015-03-24 14:15:58 -0700661 otherName := ctx.OtherModuleName(m)
662 if otherName == depNames.CrtBegin {
Colin Crossfa138792015-04-24 17:31:52 -0700663 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700664 depPaths.CrtBegin = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700665 }
666 } else if otherName == depNames.CrtEnd {
Colin Crossfa138792015-04-24 17:31:52 -0700667 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700668 depPaths.CrtEnd = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700669 }
670 } else {
Dan Albertc3144b12015-04-28 18:17:56 -0700671 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.object().outputFile())
Colin Cross21b9a242015-03-24 14:15:58 -0700672 }
673 }
674 })
675
676 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800677}
678
Colin Crossfa138792015-04-24 17:31:52 -0700679// CCLinked contains the properties and members used by libraries and executables
680type CCLinked struct {
681 CCBase
Colin Crossed4cf0b2015-03-26 14:43:45 -0700682
683 dynamicProperties struct {
Colin Cross18b6dc52015-04-28 13:20:37 -0700684 VariantIsShared bool `blueprint:"mutated"`
685 VariantIsStatic bool `blueprint:"mutated"`
686 VariantIsStaticBinary bool `blueprint:"mutated"`
Colin Crossed4cf0b2015-03-26 14:43:45 -0700687 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800688}
689
Colin Crossfa138792015-04-24 17:31:52 -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 Crossfa138792015-04-24 17:31:52 -0700695 return newCCBase(&dynamic.CCBase, module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700696}
697
Colin Crossfa138792015-04-24 17:31:52 -0700698func (c *CCLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross28d76592015-03-26 16:14:04 -0700699 if ctx.ContainsProperty("system_shared_libs") {
Colin Crossfa138792015-04-24 17:31:52 -0700700 return c.Properties.System_shared_libs
701 } else if ctx.Device() && c.Properties.Sdk_version == "" {
Colin Cross577f6e42015-03-27 18:23:34 -0700702 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 Crossfa138792015-04-24 17:31:52 -0700708func (c *CCLinked) stl(ctx common.AndroidBaseContext) string {
709 if c.Properties.Sdk_version != "" && ctx.Device() {
710 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700711 case "":
712 return "ndk_system"
713 case "c++_shared", "c++_static",
714 "stlport_shared", "stlport_static",
715 "gnustl_static":
Colin Crossfa138792015-04-24 17:31:52 -0700716 return "ndk_lib" + c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700717 default:
Colin Crossfa138792015-04-24 17:31:52 -0700718 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700719 return ""
720 }
721 }
722
Colin Crossfa138792015-04-24 17:31:52 -0700723 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700724 case "libc++", "libc++_static",
725 "stlport", "stlport_static",
726 "libstdc++":
Colin Crossfa138792015-04-24 17:31:52 -0700727 return c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700728 case "none":
729 return ""
730 case "":
Colin Cross18b6dc52015-04-28 13:20:37 -0700731 if c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700732 return "libc++_static"
Colin Cross18b6dc52015-04-28 13:20:37 -0700733 } else {
734 return "libc++" // TODO: mingw needs libstdc++
Colin Crossed4cf0b2015-03-26 14:43:45 -0700735 }
736 default:
Colin Crossfa138792015-04-24 17:31:52 -0700737 ctx.ModuleErrorf("stl: %q is not a supported STL", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700738 return ""
739 }
740}
741
Colin Cross0af4b842015-04-30 16:36:18 -0700742var hostDynamicGccLibs, hostStaticGccLibs []string
743
744func init() {
745 if runtime.GOOS == "darwin" {
746 hostDynamicGccLibs = []string{"-lc", "-lSystem"}
747 hostStaticGccLibs = []string{"NO_STATIC_HOST_BINARIES_ON_DARWIN"}
748 } else {
749 hostDynamicGccLibs = []string{"-lgcc_s", "-lgcc", "-lc", "-lgcc_s", "-lgcc"}
750 hostStaticGccLibs = []string{"-Wl,--start-group", "-lgcc", "-lgcc_eh", "-lc", "-Wl,--end-group"}
751 }
752}
Colin Cross712fc022015-04-27 11:13:34 -0700753
Colin Crosse11befc2015-04-27 17:49:17 -0700754func (c *CCLinked) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700755 stl := c.stl(ctx)
756 if ctx.Failed() {
757 return flags
758 }
759
760 switch stl {
761 case "libc++", "libc++_static":
762 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
Colin Cross28344522015-04-22 13:07:53 -0700763 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/external/libcxx/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700764 if ctx.Host() {
765 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
766 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross712fc022015-04-27 11:13:34 -0700767 flags.LdFlags = append(flags.LdFlags, "-lm", "-lpthread")
Colin Cross18b6dc52015-04-28 13:20:37 -0700768 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700769 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700770 } else {
771 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700772 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700773 }
774 case "stlport", "stlport_static":
775 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700776 flags.CFlags = append(flags.CFlags,
777 "-I${SrcDir}/external/stlport/stlport",
778 "-I${SrcDir}/bionic/libstdc++/include",
779 "-I${SrcDir}/bionic")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700780 }
781 case "libstdc++":
782 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
783 // tree is in good enough shape to not need it.
784 // Host builds will use GNU libstdc++.
785 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700786 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/bionic/libstdc++/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700787 }
788 case "ndk_system":
Colin Cross1332b002015-04-07 17:11:30 -0700789 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources/"
Colin Cross28344522015-04-22 13:07:53 -0700790 flags.CFlags = append(flags.CFlags, "-isystem "+ndkSrcRoot+"cxx-stl/system/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700791 case "ndk_libc++_shared", "ndk_libc++_static":
792 // TODO(danalbert): This really shouldn't be here...
793 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
794 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
795 // Nothing
796 case "":
797 // None or error.
798 if ctx.Host() {
799 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
800 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross18b6dc52015-04-28 13:20:37 -0700801 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700802 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700803 } else {
804 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700805 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700806 }
807 default:
Colin Crossfa138792015-04-24 17:31:52 -0700808 panic(fmt.Errorf("Unknown stl in CCLinked.Flags: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700809 }
810
811 return flags
812}
813
Colin Crosse11befc2015-04-27 17:49:17 -0700814func (c *CCLinked) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
815 depNames = c.CCBase.depNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800816
Colin Crossed4cf0b2015-03-26 14:43:45 -0700817 stl := c.stl(ctx)
818 if ctx.Failed() {
819 return depNames
820 }
821
822 switch stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700823 case "libstdc++":
824 if ctx.Device() {
825 depNames.SharedLibs = append(depNames.SharedLibs, stl)
826 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700827 case "libc++", "libc++_static":
828 if stl == "libc++" {
829 depNames.SharedLibs = append(depNames.SharedLibs, stl)
830 } else {
831 depNames.StaticLibs = append(depNames.StaticLibs, stl)
832 }
833 if ctx.Device() {
834 if ctx.Arch().ArchType == common.Arm {
835 depNames.StaticLibs = append(depNames.StaticLibs, "libunwind_llvm")
836 }
837 if c.staticBinary() {
838 depNames.StaticLibs = append(depNames.StaticLibs, "libdl")
839 } else {
840 depNames.SharedLibs = append(depNames.SharedLibs, "libdl")
841 }
842 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700843 case "stlport":
844 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
845 case "stlport_static":
846 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
847 case "":
848 // None or error.
849 case "ndk_system":
850 // TODO: Make a system STL prebuilt for the NDK.
851 // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have
Colin Crossfa138792015-04-24 17:31:52 -0700852 // its own includes. The includes are handled in CCBase.Flags().
Colin Cross577f6e42015-03-27 18:23:34 -0700853 depNames.SharedLibs = append([]string{"libstdc++"}, depNames.SharedLibs...)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700854 case "ndk_libc++_shared", "ndk_libstlport_shared":
855 depNames.SharedLibs = append(depNames.SharedLibs, stl)
856 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
857 depNames.StaticLibs = append(depNames.StaticLibs, stl)
858 default:
Colin Crosse11befc2015-04-27 17:49:17 -0700859 panic(fmt.Errorf("Unknown stl in CCLinked.depNames: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700860 }
861
Colin Cross74d1ec02015-04-28 13:30:13 -0700862 if ctx.ModuleName() != "libcompiler_rt-extras" {
863 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
864 }
865
Colin Crossf6566ed2015-03-24 11:13:38 -0700866 if ctx.Device() {
Colin Cross77b00fa2015-03-16 16:15:49 -0700867 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700868 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700869
Colin Cross18b6dc52015-04-28 13:20:37 -0700870 if !c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700871 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
872 }
Colin Cross577f6e42015-03-27 18:23:34 -0700873
Colin Crossfa138792015-04-24 17:31:52 -0700874 if c.Properties.Sdk_version != "" {
875 version := c.Properties.Sdk_version
Colin Cross577f6e42015-03-27 18:23:34 -0700876 depNames.SharedLibs = append(depNames.SharedLibs,
877 "ndk_libc."+version,
878 "ndk_libm."+version,
879 )
880 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800881 }
882
Colin Cross21b9a242015-03-24 14:15:58 -0700883 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800884}
885
Colin Crossed4cf0b2015-03-26 14:43:45 -0700886// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
887type ccLinkedInterface interface {
888 // Returns true if the build options for the module have selected a static or shared build
889 buildStatic() bool
890 buildShared() bool
891
892 // Sets whether a specific variant is static or shared
Colin Cross18b6dc52015-04-28 13:20:37 -0700893 setStatic(bool)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700894
Colin Cross18b6dc52015-04-28 13:20:37 -0700895 // Returns whether a specific variant is a static library or binary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700896 static() bool
Colin Cross18b6dc52015-04-28 13:20:37 -0700897
898 // Returns whether a module is a static binary
899 staticBinary() bool
Colin Crossed4cf0b2015-03-26 14:43:45 -0700900}
901
902var _ ccLinkedInterface = (*CCLibrary)(nil)
903var _ ccLinkedInterface = (*CCBinary)(nil)
904
Colin Crossfa138792015-04-24 17:31:52 -0700905func (c *CCLinked) static() bool {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700906 return c.dynamicProperties.VariantIsStatic
907}
908
Colin Cross18b6dc52015-04-28 13:20:37 -0700909func (c *CCLinked) staticBinary() bool {
910 return c.dynamicProperties.VariantIsStaticBinary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700911}
912
Colin Cross18b6dc52015-04-28 13:20:37 -0700913func (c *CCLinked) setStatic(static bool) {
914 c.dynamicProperties.VariantIsStatic = static
Colin Crossed4cf0b2015-03-26 14:43:45 -0700915}
916
Colin Cross28344522015-04-22 13:07:53 -0700917type ccExportedFlagsProducer interface {
918 exportedFlags() []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800919}
920
921//
922// Combined static+shared libraries
923//
924
Colin Cross97ba0732015-03-23 17:50:24 -0700925type CCLibrary struct {
Colin Crossfa138792015-04-24 17:31:52 -0700926 CCLinked
Colin Cross3f40fa42015-01-30 17:27:36 -0800927
Colin Cross28344522015-04-22 13:07:53 -0700928 reuseFrom ccLibraryInterface
929 reuseObjFiles []string
930 objFiles []string
931 exportFlags []string
932 out string
Colin Cross3f40fa42015-01-30 17:27:36 -0800933
Colin Cross97ba0732015-03-23 17:50:24 -0700934 LibraryProperties struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800935 BuildStatic bool `blueprint:"mutated"`
936 BuildShared bool `blueprint:"mutated"`
Colin Crossed4cf0b2015-03-26 14:43:45 -0700937 Static struct {
Colin Cross2732e9a2015-04-28 13:23:52 -0700938 Srcs []string `android:"arch_variant"`
939 Cflags []string `android:"arch_variant"`
940 Whole_static_libs []string `android:"arch_variant"`
941 Static_libs []string `android:"arch_variant"`
942 Shared_libs []string `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800943 } `android:"arch_variant"`
944 Shared struct {
Colin Cross2732e9a2015-04-28 13:23:52 -0700945 Srcs []string `android:"arch_variant"`
946 Cflags []string `android:"arch_variant"`
947 Whole_static_libs []string `android:"arch_variant"`
948 Static_libs []string `android:"arch_variant"`
949 Shared_libs []string `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800950 } `android:"arch_variant"`
951 }
952}
953
Colin Crossed4cf0b2015-03-26 14:43:45 -0700954func (c *CCLibrary) buildStatic() bool {
955 return c.LibraryProperties.BuildStatic
956}
957
958func (c *CCLibrary) buildShared() bool {
959 return c.LibraryProperties.BuildShared
960}
961
Colin Cross97ba0732015-03-23 17:50:24 -0700962type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700963 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -0700964 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700965 setReuseFrom(ccLibraryInterface)
966 getReuseFrom() ccLibraryInterface
967 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -0700968 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -0700969}
970
Colin Crossed4cf0b2015-03-26 14:43:45 -0700971var _ ccLibraryInterface = (*CCLibrary)(nil)
972
Colin Cross97ba0732015-03-23 17:50:24 -0700973func (c *CCLibrary) ccLibrary() *CCLibrary {
974 return c
Colin Cross3f40fa42015-01-30 17:27:36 -0800975}
976
Colin Cross97ba0732015-03-23 17:50:24 -0700977func NewCCLibrary(library *CCLibrary, module CCModuleType,
978 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
979
Colin Crossfa138792015-04-24 17:31:52 -0700980 return newCCDynamic(&library.CCLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -0700981 &library.LibraryProperties)
982}
983
984func CCLibraryFactory() (blueprint.Module, []interface{}) {
985 module := &CCLibrary{}
986
987 module.LibraryProperties.BuildShared = true
988 module.LibraryProperties.BuildStatic = true
989
990 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
991}
992
Colin Cross0676e2d2015-04-24 17:39:18 -0700993func (c *CCLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -0700994 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Cross2732e9a2015-04-28 13:23:52 -0700995 if c.static() {
996 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Static.Whole_static_libs...)
997 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Static.Static_libs...)
998 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Static.Shared_libs...)
999 } else {
Colin Crossf6566ed2015-03-24 11:13:38 -07001000 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001001 if c.Properties.Sdk_version == "" {
1002 depNames.CrtBegin = "crtbegin_so"
1003 depNames.CrtEnd = "crtend_so"
1004 } else {
1005 depNames.CrtBegin = "ndk_crtbegin_so." + c.Properties.Sdk_version
1006 depNames.CrtEnd = "ndk_crtend_so." + c.Properties.Sdk_version
1007 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001008 }
Colin Cross2732e9a2015-04-28 13:23:52 -07001009 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Shared.Whole_static_libs...)
1010 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Shared.Static_libs...)
1011 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Shared.Shared_libs...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001012 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001013
Colin Cross21b9a242015-03-24 14:15:58 -07001014 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001015}
1016
Colin Cross97ba0732015-03-23 17:50:24 -07001017func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001018 return c.out
1019}
1020
Colin Crossed4cf0b2015-03-26 14:43:45 -07001021func (c *CCLibrary) getReuseObjFiles() []string {
1022 return c.reuseObjFiles
1023}
1024
1025func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
1026 c.reuseFrom = reuseFrom
1027}
1028
1029func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
1030 return c.reuseFrom
1031}
1032
Colin Cross97ba0732015-03-23 17:50:24 -07001033func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001034 return c.objFiles
1035}
1036
Colin Cross28344522015-04-22 13:07:53 -07001037func (c *CCLibrary) exportedFlags() []string {
1038 return c.exportFlags
Colin Cross3f40fa42015-01-30 17:27:36 -08001039}
1040
Colin Cross0676e2d2015-04-24 17:39:18 -07001041func (c *CCLibrary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001042 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001043
Colin Cross97ba0732015-03-23 17:50:24 -07001044 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -08001045
Colin Crossd8e780d2015-04-28 17:39:43 -07001046 if c.static() {
1047 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Static.Cflags...)
1048 } else {
1049 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Shared.Cflags...)
1050 }
1051
Colin Cross18b6dc52015-04-28 13:20:37 -07001052 if !c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001053 libName := ctx.ModuleName()
1054 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
1055 sharedFlag := "-Wl,-shared"
Colin Crossfa138792015-04-24 17:31:52 -07001056 if c.Properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001057 sharedFlag = "-shared"
1058 }
Colin Crossf6566ed2015-03-24 11:13:38 -07001059 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001060 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001061 }
Colin Cross97ba0732015-03-23 17:50:24 -07001062
Colin Cross0af4b842015-04-30 16:36:18 -07001063 if ctx.Darwin() {
1064 flags.LdFlags = append(flags.LdFlags,
1065 "-dynamiclib",
1066 "-single_module",
1067 //"-read_only_relocs suppress",
1068 "-install_name @rpath/"+libName+sharedLibraryExtension,
1069 )
1070 } else {
1071 flags.LdFlags = append(flags.LdFlags,
1072 "-Wl,--gc-sections",
1073 sharedFlag,
1074 "-Wl,-soname,"+libName+sharedLibraryExtension,
1075 )
1076 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001077 }
Colin Cross97ba0732015-03-23 17:50:24 -07001078
1079 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001080}
1081
Colin Cross97ba0732015-03-23 17:50:24 -07001082func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1083 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001084
1085 staticFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001086 objFilesStatic := c.customCompileObjs(ctx, staticFlags, common.DeviceStaticLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001087 c.LibraryProperties.Static.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001088
1089 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001090 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001091
1092 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1093
Colin Cross0af4b842015-04-30 16:36:18 -07001094 if ctx.Darwin() {
1095 TransformDarwinObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1096 } else {
1097 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1098 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001099
1100 c.objFiles = objFiles
1101 c.out = outputFile
Colin Crossf2298272015-05-12 11:36:53 -07001102
1103 common.CheckModuleSrcDirsExist(ctx, c.Properties.Export_include_dirs, "export_include_dirs")
Colin Crossfa138792015-04-24 17:31:52 -07001104 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001105 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001106
1107 ctx.CheckbuildFile(outputFile)
1108}
1109
Colin Cross97ba0732015-03-23 17:50:24 -07001110func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1111 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001112
1113 sharedFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001114 objFilesShared := c.customCompileObjs(ctx, sharedFlags, common.DeviceSharedLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001115 c.LibraryProperties.Shared.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001116
1117 objFiles = append(objFiles, objFilesShared...)
1118
1119 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1120
Colin Cross97ba0732015-03-23 17:50:24 -07001121 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001122 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, false,
Colin Cross77b00fa2015-03-16 16:15:49 -07001123 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001124
1125 c.out = outputFile
Colin Crossfa138792015-04-24 17:31:52 -07001126 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001127 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001128}
1129
Colin Cross97ba0732015-03-23 17:50:24 -07001130func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1131 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001132
1133 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001134 if c.getReuseFrom().ccLibrary() == c {
1135 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001136 } else {
Colin Cross2732e9a2015-04-28 13:23:52 -07001137 if c.getReuseFrom().ccLibrary().LibraryProperties.Static.Cflags == nil &&
1138 c.LibraryProperties.Shared.Cflags == nil {
1139 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
1140 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001141 }
1142
Colin Crossed4cf0b2015-03-26 14:43:45 -07001143 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001144 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1145 } else {
1146 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1147 }
1148}
1149
Colin Cross97ba0732015-03-23 17:50:24 -07001150func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001151 // Static libraries do not get installed.
1152}
1153
Colin Cross97ba0732015-03-23 17:50:24 -07001154func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001155 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001156 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001157 installDir = "lib64"
1158 }
1159
Colin Crossfa138792015-04-24 17:31:52 -07001160 ctx.InstallFile(filepath.Join(installDir, c.Properties.Relative_install_path), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001161}
1162
Colin Cross97ba0732015-03-23 17:50:24 -07001163func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001164 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001165 c.installStaticLibrary(ctx, flags)
1166 } else {
1167 c.installSharedLibrary(ctx, flags)
1168 }
1169}
1170
Colin Cross3f40fa42015-01-30 17:27:36 -08001171//
1172// Objects (for crt*.o)
1173//
1174
Dan Albertc3144b12015-04-28 18:17:56 -07001175type ccObjectProvider interface {
1176 object() *ccObject
1177}
1178
Colin Cross3f40fa42015-01-30 17:27:36 -08001179type ccObject struct {
Colin Crossfa138792015-04-24 17:31:52 -07001180 CCBase
Colin Cross3f40fa42015-01-30 17:27:36 -08001181 out string
1182}
1183
Dan Albertc3144b12015-04-28 18:17:56 -07001184func (c *ccObject) object() *ccObject {
1185 return c
1186}
1187
Colin Cross97ba0732015-03-23 17:50:24 -07001188func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001189 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001190
Colin Crossfa138792015-04-24 17:31:52 -07001191 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001192}
1193
1194func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1195 // object files can't have any dynamic dependencies
1196 return nil
1197}
1198
Colin Cross0676e2d2015-04-24 17:39:18 -07001199func (*ccObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001200 // object files can't have any dynamic dependencies
1201 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001202}
1203
1204func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001205 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001206
Colin Cross97ba0732015-03-23 17:50:24 -07001207 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001208
1209 var outputFile string
1210 if len(objFiles) == 1 {
1211 outputFile = objFiles[0]
1212 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001213 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+objectExtension)
Colin Cross3f40fa42015-01-30 17:27:36 -08001214 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1215 }
1216
1217 c.out = outputFile
1218
1219 ctx.CheckbuildFile(outputFile)
1220}
1221
Colin Cross97ba0732015-03-23 17:50:24 -07001222func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001223 // Object files do not get installed.
1224}
1225
Colin Cross3f40fa42015-01-30 17:27:36 -08001226func (c *ccObject) outputFile() string {
1227 return c.out
1228}
1229
Dan Albertc3144b12015-04-28 18:17:56 -07001230var _ ccObjectProvider = (*ccObject)(nil)
1231
Colin Cross3f40fa42015-01-30 17:27:36 -08001232//
1233// Executables
1234//
1235
Colin Cross97ba0732015-03-23 17:50:24 -07001236type CCBinary struct {
Colin Crossfa138792015-04-24 17:31:52 -07001237 CCLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001238 out string
Colin Crossd350ecd2015-04-28 13:25:36 -07001239 installFile string
Colin Cross97ba0732015-03-23 17:50:24 -07001240 BinaryProperties struct {
1241 // static_executable: compile executable with -static
1242 Static_executable bool
1243
1244 // stem: set the name of the output
1245 Stem string `android:"arch_variant"`
1246
Colin Cross4ae185c2015-03-26 15:12:10 -07001247 // suffix: append to the name of the output
1248 Suffix string `android:"arch_variant"`
1249
Colin Cross97ba0732015-03-23 17:50:24 -07001250 // prefix_symbols: if set, add an extra objcopy --prefix-symbols= step
1251 Prefix_symbols string
1252 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001253}
1254
Colin Crossed4cf0b2015-03-26 14:43:45 -07001255func (c *CCBinary) buildStatic() bool {
1256 return c.BinaryProperties.Static_executable
1257}
1258
1259func (c *CCBinary) buildShared() bool {
1260 return !c.BinaryProperties.Static_executable
1261}
1262
Colin Cross97ba0732015-03-23 17:50:24 -07001263func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001264 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001265 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001266 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001267 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001268
1269 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001270}
1271
Colin Cross0676e2d2015-04-24 17:39:18 -07001272func (c *CCBinary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001273 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001274 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001275 if c.Properties.Sdk_version == "" {
1276 if c.BinaryProperties.Static_executable {
1277 depNames.CrtBegin = "crtbegin_static"
1278 } else {
1279 depNames.CrtBegin = "crtbegin_dynamic"
1280 }
1281 depNames.CrtEnd = "crtend_android"
Colin Cross3f40fa42015-01-30 17:27:36 -08001282 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001283 if c.BinaryProperties.Static_executable {
1284 depNames.CrtBegin = "ndk_crtbegin_static." + c.Properties.Sdk_version
1285 } else {
1286 depNames.CrtBegin = "ndk_crtbegin_dynamic." + c.Properties.Sdk_version
1287 }
1288 depNames.CrtEnd = "ndk_crtend_android." + c.Properties.Sdk_version
Colin Cross3f40fa42015-01-30 17:27:36 -08001289 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001290
1291 if c.BinaryProperties.Static_executable {
Colin Cross74d1ec02015-04-28 13:30:13 -07001292 if c.stl(ctx) == "libc++_static" {
1293 depNames.StaticLibs = append(depNames.StaticLibs, "libm", "libc", "libdl")
1294 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001295 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1296 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1297 // move them to the beginning of deps.LateStaticLibs
1298 var groupLibs []string
1299 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1300 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1301 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1302 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001303 }
Colin Cross21b9a242015-03-24 14:15:58 -07001304 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001305}
1306
Colin Cross97ba0732015-03-23 17:50:24 -07001307func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001308 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001309
Colin Cross1f8f2342015-03-26 16:09:47 -07001310 props = append(props, &binary.BinaryProperties)
1311
Colin Crossfa138792015-04-24 17:31:52 -07001312 return newCCDynamic(&binary.CCLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001313}
1314
Colin Cross97ba0732015-03-23 17:50:24 -07001315func CCBinaryFactory() (blueprint.Module, []interface{}) {
1316 module := &CCBinary{}
1317
1318 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001319}
1320
Colin Cross18b6dc52015-04-28 13:20:37 -07001321func (c *CCBinary) ModifyProperties(ctx common.AndroidBaseContext) {
Colin Cross0af4b842015-04-30 16:36:18 -07001322 if ctx.Darwin() {
1323 c.BinaryProperties.Static_executable = false
1324 }
Colin Cross18b6dc52015-04-28 13:20:37 -07001325 if c.BinaryProperties.Static_executable {
1326 c.dynamicProperties.VariantIsStaticBinary = true
1327 }
1328}
1329
Colin Cross0676e2d2015-04-24 17:39:18 -07001330func (c *CCBinary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001331 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001332
Colin Cross97ba0732015-03-23 17:50:24 -07001333 flags.CFlags = append(flags.CFlags, "-fpie")
1334
Colin Crossf6566ed2015-03-24 11:13:38 -07001335 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001336 if c.BinaryProperties.Static_executable {
1337 // Clang driver needs -static to create static executable.
1338 // However, bionic/linker uses -shared to overwrite.
1339 // Linker for x86 targets does not allow coexistance of -static and -shared,
1340 // so we add -static only if -shared is not used.
1341 if !inList("-shared", flags.LdFlags) {
1342 flags.LdFlags = append(flags.LdFlags, "-static")
1343 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001344
Colin Crossed4cf0b2015-03-26 14:43:45 -07001345 flags.LdFlags = append(flags.LdFlags,
1346 "-nostdlib",
1347 "-Bstatic",
1348 "-Wl,--gc-sections",
1349 )
1350
1351 } else {
1352 linker := "/system/bin/linker"
1353 if flags.Toolchain.Is64Bit() {
1354 linker = "/system/bin/linker64"
1355 }
1356
1357 flags.LdFlags = append(flags.LdFlags,
1358 "-nostdlib",
1359 "-Bdynamic",
1360 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1361 "-Wl,--gc-sections",
1362 "-Wl,-z,nocopyreloc",
1363 )
1364 }
Colin Cross0af4b842015-04-30 16:36:18 -07001365 } else if ctx.Darwin() {
1366 flags.LdFlags = append(flags.LdFlags, "-Wl,-headerpad_max_install_names")
Colin Cross3f40fa42015-01-30 17:27:36 -08001367 }
1368
Colin Cross97ba0732015-03-23 17:50:24 -07001369 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001370}
1371
Colin Cross97ba0732015-03-23 17:50:24 -07001372func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1373 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001374
Colin Crossfa138792015-04-24 17:31:52 -07001375 if !c.BinaryProperties.Static_executable && inList("libc", c.Properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001376 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1377 "from static libs or set static_executable: true")
1378 }
1379
1380 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001381 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001382 if c.BinaryProperties.Prefix_symbols != "" {
1383 afterPrefixSymbols := outputFile
1384 outputFile = outputFile + ".intermediate"
1385 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1386 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1387 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001388
Colin Cross97ba0732015-03-23 17:50:24 -07001389 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001390 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001391 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001392}
Colin Cross3f40fa42015-01-30 17:27:36 -08001393
Colin Cross97ba0732015-03-23 17:50:24 -07001394func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossd350ecd2015-04-28 13:25:36 -07001395 c.installFile = ctx.InstallFile(filepath.Join("bin", c.Properties.Relative_install_path), c.out)
1396}
1397
1398func (c *CCBinary) HostToolPath() string {
1399 if c.HostOrDevice().Host() {
1400 return c.installFile
1401 }
1402 return ""
Dan Albertc403f7c2015-03-18 14:01:18 -07001403}
1404
Colin Cross9ffb4f52015-04-24 17:48:09 -07001405type CCTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001406 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001407
Colin Cross9ffb4f52015-04-24 17:48:09 -07001408 TestProperties struct {
Colin Cross6b290692015-03-19 14:05:33 -07001409 // test_per_src: Create a separate test for each source file. Useful when there is
1410 // global state that can not be torn down and reset between each test suite.
1411 Test_per_src bool
1412 }
Dan Albertc403f7c2015-03-18 14:01:18 -07001413}
1414
Colin Cross9ffb4f52015-04-24 17:48:09 -07001415func (c *CCTest) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross0676e2d2015-04-24 17:39:18 -07001416 flags = c.CCBinary.flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001417
Colin Cross97ba0732015-03-23 17:50:24 -07001418 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001419 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001420 flags.CFlags = append(flags.CFlags, "-O0", "-g")
Colin Cross28344522015-04-22 13:07:53 -07001421 flags.LdFlags = append(flags.LdFlags, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001422 }
1423
1424 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross28344522015-04-22 13:07:53 -07001425 flags.CFlags = append(flags.CFlags,
1426 "-I"+filepath.Join(ctx.AConfig().SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001427
Colin Cross21b9a242015-03-24 14:15:58 -07001428 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001429}
1430
Colin Cross9ffb4f52015-04-24 17:48:09 -07001431func (c *CCTest) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001432 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
Colin Crossa8a93d32015-04-28 13:26:49 -07001433 depNames = c.CCBinary.depNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -07001434 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001435}
1436
Colin Cross9ffb4f52015-04-24 17:48:09 -07001437func (c *CCTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001438 if ctx.Device() {
Colin Crossa8a93d32015-04-28 13:26:49 -07001439 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001440 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001441 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001442 }
1443}
1444
Colin Cross9ffb4f52015-04-24 17:48:09 -07001445func (c *CCTest) testPerSrc() bool {
1446 return c.TestProperties.Test_per_src
Colin Cross6b290692015-03-19 14:05:33 -07001447}
1448
Colin Cross9ffb4f52015-04-24 17:48:09 -07001449func (c *CCTest) test() *CCTest {
1450 return c
1451}
1452
1453func NewCCTest(test *CCTest, module CCModuleType,
1454 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1455
1456 props = append(props, &test.TestProperties)
1457
1458 return NewCCBinary(&test.CCBinary, module, hod, props...)
1459}
1460
1461func CCTestFactory() (blueprint.Module, []interface{}) {
1462 module := &CCTest{}
1463
1464 return NewCCTest(module, module, common.HostAndDeviceSupported)
1465}
1466
1467type testPerSrc interface {
1468 test() *CCTest
1469 testPerSrc() bool
1470}
1471
1472var _ testPerSrc = (*CCTest)(nil)
1473
Colin Cross6b290692015-03-19 14:05:33 -07001474func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001475 if test, ok := mctx.Module().(testPerSrc); ok {
1476 if test.testPerSrc() {
1477 testNames := make([]string, len(test.test().Properties.Srcs))
1478 for i, src := range test.test().Properties.Srcs {
Colin Cross6b290692015-03-19 14:05:33 -07001479 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1480 }
1481 tests := mctx.CreateLocalVariations(testNames...)
Colin Cross9ffb4f52015-04-24 17:48:09 -07001482 for i, src := range test.test().Properties.Srcs {
1483 tests[i].(testPerSrc).test().Properties.Srcs = []string{src}
1484 tests[i].(testPerSrc).test().BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001485 }
1486 }
1487 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001488}
1489
Colin Cross2ba19d92015-05-07 15:44:20 -07001490type CCBenchmark struct {
1491 CCBinary
1492}
1493
1494func (c *CCBenchmark) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1495 depNames = c.CCBinary.depNames(ctx, depNames)
1496 depNames.StaticLibs = append(depNames.StaticLibs, "libbenchmark")
1497 return depNames
1498}
1499
1500func (c *CCBenchmark) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1501 if ctx.Device() {
1502 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
1503 } else {
1504 c.CCBinary.installModule(ctx, flags)
1505 }
1506}
1507
1508func NewCCBenchmark(test *CCBenchmark, module CCModuleType,
1509 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1510
1511 return NewCCBinary(&test.CCBinary, module, hod, props...)
1512}
1513
1514func CCBenchmarkFactory() (blueprint.Module, []interface{}) {
1515 module := &CCBenchmark{}
1516
1517 return NewCCBenchmark(module, module, common.HostAndDeviceSupported)
1518}
1519
Colin Cross3f40fa42015-01-30 17:27:36 -08001520//
1521// Static library
1522//
1523
Colin Cross97ba0732015-03-23 17:50:24 -07001524func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1525 module := &CCLibrary{}
1526 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001527
Colin Cross97ba0732015-03-23 17:50:24 -07001528 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001529}
1530
1531//
1532// Shared libraries
1533//
1534
Colin Cross97ba0732015-03-23 17:50:24 -07001535func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1536 module := &CCLibrary{}
1537 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001538
Colin Cross97ba0732015-03-23 17:50:24 -07001539 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001540}
1541
1542//
1543// Host static library
1544//
1545
Colin Cross97ba0732015-03-23 17:50:24 -07001546func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1547 module := &CCLibrary{}
1548 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001549
Colin Cross97ba0732015-03-23 17:50:24 -07001550 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001551}
1552
1553//
1554// Host Shared libraries
1555//
1556
Colin Cross97ba0732015-03-23 17:50:24 -07001557func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1558 module := &CCLibrary{}
1559 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001560
Colin Cross97ba0732015-03-23 17:50:24 -07001561 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001562}
1563
1564//
1565// Host Binaries
1566//
1567
Colin Cross97ba0732015-03-23 17:50:24 -07001568func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1569 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001570
Colin Cross97ba0732015-03-23 17:50:24 -07001571 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001572}
1573
1574//
Colin Cross1f8f2342015-03-26 16:09:47 -07001575// Host Tests
1576//
1577
1578func CCTestHostFactory() (blueprint.Module, []interface{}) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001579 module := &CCTest{}
Colin Cross1f8f2342015-03-26 16:09:47 -07001580 return NewCCBinary(&module.CCBinary, module, common.HostSupported,
Colin Cross9ffb4f52015-04-24 17:48:09 -07001581 &module.TestProperties)
Colin Cross1f8f2342015-03-26 16:09:47 -07001582}
1583
1584//
Colin Cross2ba19d92015-05-07 15:44:20 -07001585// Host Benchmarks
1586//
1587
1588func CCBenchmarkHostFactory() (blueprint.Module, []interface{}) {
1589 module := &CCBenchmark{}
1590 return NewCCBinary(&module.CCBinary, module, common.HostSupported)
1591}
1592
1593//
Colin Cross3f40fa42015-01-30 17:27:36 -08001594// Device libraries shipped with gcc
1595//
1596
1597type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001598 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001599}
1600
1601func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1602 // toolchain libraries can't have any dependencies
1603 return nil
1604}
1605
Colin Cross0676e2d2015-04-24 17:39:18 -07001606func (*toolchainLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001607 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001608 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001609}
1610
Colin Cross97ba0732015-03-23 17:50:24 -07001611func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001612 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001613
Colin Cross97ba0732015-03-23 17:50:24 -07001614 module.LibraryProperties.BuildStatic = true
1615
Colin Crossfa138792015-04-24 17:31:52 -07001616 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth,
Colin Cross21b9a242015-03-24 14:15:58 -07001617 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001618}
1619
1620func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001621 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001622
1623 libName := ctx.ModuleName() + staticLibraryExtension
1624 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1625
1626 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1627
1628 c.out = outputFile
1629
1630 ctx.CheckbuildFile(outputFile)
1631}
1632
Colin Cross97ba0732015-03-23 17:50:24 -07001633func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001634 // Toolchain libraries do not get installed.
1635}
1636
Dan Albertbe961682015-03-18 23:38:50 -07001637// NDK prebuilt libraries.
1638//
1639// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1640// either (with the exception of the shared STLs, which are installed to the app's directory rather
1641// than to the system image).
1642
1643func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1644 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
Colin Cross1332b002015-04-07 17:11:30 -07001645 ctx.AConfig().SrcDir(), version, toolchain.Name())
Dan Albertbe961682015-03-18 23:38:50 -07001646}
1647
Dan Albertc3144b12015-04-28 18:17:56 -07001648func ndkPrebuiltModuleToPath(ctx common.AndroidModuleContext, toolchain Toolchain,
1649 ext string, version string) string {
1650
1651 // NDK prebuilts are named like: ndk_NAME.EXT.SDK_VERSION.
1652 // We want to translate to just NAME.EXT
1653 name := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1654 dir := getNdkLibDir(ctx, toolchain, version)
1655 return filepath.Join(dir, name+ext)
1656}
1657
1658type ndkPrebuiltObject struct {
1659 ccObject
1660}
1661
1662func (*ndkPrebuiltObject) AndroidDynamicDependencies(
1663 ctx common.AndroidDynamicDependerModuleContext) []string {
1664
1665 // NDK objects can't have any dependencies
1666 return nil
1667}
1668
1669func (*ndkPrebuiltObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1670 // NDK objects can't have any dependencies
1671 return CCDeps{}
1672}
1673
1674func NdkPrebuiltObjectFactory() (blueprint.Module, []interface{}) {
1675 module := &ndkPrebuiltObject{}
1676 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
1677}
1678
1679func (c *ndkPrebuiltObject) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1680 deps CCDeps, objFiles []string) {
1681 // A null build step, but it sets up the output path.
1682 if !strings.HasPrefix(ctx.ModuleName(), "ndk_crt") {
1683 ctx.ModuleErrorf("NDK prebuilts must have an ndk_crt prefixed name")
1684 }
1685
1686 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, objectExtension, c.Properties.Sdk_version)
1687}
1688
1689func (c *ndkPrebuiltObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1690 // Objects do not get installed.
1691}
1692
1693var _ ccObjectProvider = (*ndkPrebuiltObject)(nil)
1694
Dan Albertbe961682015-03-18 23:38:50 -07001695type ndkPrebuiltLibrary struct {
1696 CCLibrary
1697}
1698
1699func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1700 ctx common.AndroidDynamicDependerModuleContext) []string {
1701
1702 // NDK libraries can't have any dependencies
1703 return nil
1704}
1705
Colin Cross0676e2d2015-04-24 17:39:18 -07001706func (*ndkPrebuiltLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Dan Albertbe961682015-03-18 23:38:50 -07001707 // NDK libraries can't have any dependencies
1708 return CCDeps{}
1709}
1710
1711func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1712 module := &ndkPrebuiltLibrary{}
1713 module.LibraryProperties.BuildShared = true
1714 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1715}
1716
1717func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1718 deps CCDeps, objFiles []string) {
1719 // A null build step, but it sets up the output path.
1720 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1721 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1722 }
1723
Colin Crossfa138792015-04-24 17:31:52 -07001724 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001725 c.exportFlags = []string{common.JoinWithPrefix(includeDirs, "-isystem ")}
Dan Albertbe961682015-03-18 23:38:50 -07001726
Dan Albertc3144b12015-04-28 18:17:56 -07001727 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, sharedLibraryExtension,
1728 c.Properties.Sdk_version)
Dan Albertbe961682015-03-18 23:38:50 -07001729}
1730
1731func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc3144b12015-04-28 18:17:56 -07001732 // NDK prebuilt libraries do not get installed.
Dan Albertbe961682015-03-18 23:38:50 -07001733}
1734
1735// The NDK STLs are slightly different from the prebuilt system libraries:
1736// * Are not specific to each platform version.
1737// * The libraries are not in a predictable location for each STL.
1738
1739type ndkPrebuiltStl struct {
1740 ndkPrebuiltLibrary
1741}
1742
1743type ndkPrebuiltStaticStl struct {
1744 ndkPrebuiltStl
1745}
1746
1747type ndkPrebuiltSharedStl struct {
1748 ndkPrebuiltStl
1749}
1750
1751func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1752 module := &ndkPrebuiltSharedStl{}
1753 module.LibraryProperties.BuildShared = true
1754 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1755}
1756
1757func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1758 module := &ndkPrebuiltStaticStl{}
1759 module.LibraryProperties.BuildStatic = true
1760 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1761}
1762
1763func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1764 gccVersion := toolchain.GccVersion()
1765 var libDir string
1766 switch stl {
1767 case "libstlport":
1768 libDir = "cxx-stl/stlport/libs"
1769 case "libc++":
1770 libDir = "cxx-stl/llvm-libc++/libs"
1771 case "libgnustl":
1772 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1773 }
1774
1775 if libDir != "" {
Colin Cross1332b002015-04-07 17:11:30 -07001776 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources"
Dan Albertbe961682015-03-18 23:38:50 -07001777 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1778 }
1779
1780 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1781 return ""
1782}
1783
1784func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1785 deps CCDeps, objFiles []string) {
1786 // A null build step, but it sets up the output path.
1787 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1788 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1789 }
1790
Colin Crossfa138792015-04-24 17:31:52 -07001791 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001792 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Dan Albertbe961682015-03-18 23:38:50 -07001793
1794 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1795 libExt := sharedLibraryExtension
1796 if c.LibraryProperties.BuildStatic {
1797 libExt = staticLibraryExtension
1798 }
1799
1800 stlName := strings.TrimSuffix(libName, "_shared")
1801 stlName = strings.TrimSuffix(stlName, "_static")
1802 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1803 c.out = libDir + "/" + libName + libExt
1804}
1805
Colin Cross3f40fa42015-01-30 17:27:36 -08001806func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001807 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001808 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001809 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001810 modules = mctx.CreateLocalVariations("static", "shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001811 modules[0].(ccLinkedInterface).setStatic(true)
1812 modules[1].(ccLinkedInterface).setStatic(false)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001813 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001814 modules = mctx.CreateLocalVariations("static")
Colin Cross18b6dc52015-04-28 13:20:37 -07001815 modules[0].(ccLinkedInterface).setStatic(true)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001816 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001817 modules = mctx.CreateLocalVariations("shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001818 modules[0].(ccLinkedInterface).setStatic(false)
Colin Cross3f40fa42015-01-30 17:27:36 -08001819 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001820 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001821 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001822
1823 if _, ok := c.(ccLibraryInterface); ok {
1824 reuseFrom := modules[0].(ccLibraryInterface)
1825 for _, m := range modules {
1826 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001827 }
1828 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001829 }
1830}
Colin Cross74d1ec02015-04-28 13:30:13 -07001831
1832// lastUniqueElements returns all unique elements of a slice, keeping the last copy of each
1833// modifies the slice contents in place, and returns a subslice of the original slice
1834func lastUniqueElements(list []string) []string {
1835 totalSkip := 0
1836 for i := len(list) - 1; i >= totalSkip; i-- {
1837 skip := 0
1838 for j := i - 1; j >= totalSkip; j-- {
1839 if list[i] == list[j] {
1840 skip++
1841 } else {
1842 list[j+skip] = list[j]
1843 }
1844 }
1845 totalSkip += skip
1846 }
1847 return list[totalSkip:]
1848}