blob: 1bef029f3a628476fdc14c0fd1cc2fc88ce8f579 [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 Cross7d5136f2015-05-11 13:39:40 -0700154// Properties used to compile all C or C++ modules
155type CCBaseProperties struct {
156 // list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
157 Srcs []string `android:"arch_variant,arch_subtract"`
158
159 // list of module-specific flags that will be used for C and C++ compiles.
160 Cflags []string `android:"arch_variant"`
161
162 // list of module-specific flags that will be used for C++ compiles
163 Cppflags []string `android:"arch_variant"`
164
165 // list of module-specific flags that will be used for C compiles
166 Conlyflags []string `android:"arch_variant"`
167
168 // list of module-specific flags that will be used for .S compiles
169 Asflags []string `android:"arch_variant"`
170
171 // list of module-specific flags that will be used for .y and .yy compiles
172 Yaccflags []string
173
174 // list of module-specific flags that will be used for all link steps
175 Ldflags []string `android:"arch_variant"`
176
177 // the instruction set architecture to use to compile the C/C++
178 // module.
179 Instruction_set string `android:"arch_variant"`
180
181 // list of directories relative to the root of the source tree that will
182 // be added to the include path using -I.
183 // If possible, don't use this. If adding paths from the current directory use
184 // local_include_dirs, if adding paths from other modules use export_include_dirs in
185 // that module.
186 Include_dirs []string `android:"arch_variant"`
187
188 // list of directories relative to the Blueprints file that will
189 // be added to the include path using -I
190 Local_include_dirs []string `android:"arch_variant"`
191
192 // list of directories relative to the Blueprints file that will
193 // be added to the include path using -I for any module that links against this module
194 Export_include_dirs []string `android:"arch_variant"`
195
196 // list of module-specific flags that will be used for C and C++ compiles when
197 // compiling with clang
198 Clang_cflags []string `android:"arch_variant"`
199
200 // list of module-specific flags that will be used for .S compiles when
201 // compiling with clang
202 Clang_asflags []string `android:"arch_variant"`
203
204 // list of system libraries that will be dynamically linked to
205 // shared library and executable modules. If unset, generally defaults to libc
206 // and libm. Set to [] to prevent linking against libc and libm.
207 System_shared_libs []string
208
209 // list of modules whose object files should be linked into this module
210 // in their entirety. For static library modules, all of the .o files from the intermediate
211 // directory of the dependency will be linked into this modules .a file. For a shared library,
212 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
213 Whole_static_libs []string `android:"arch_variant"`
214
215 // list of modules that should be statically linked into this module.
216 Static_libs []string `android:"arch_variant"`
217
218 // list of modules that should be dynamically linked into this module.
219 Shared_libs []string `android:"arch_variant"`
220
221 // allow the module to contain undefined symbols. By default,
222 // modules cannot contain undefined symbols that are not satisified by their immediate
223 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
224 // This flag should only be necessary for compiling low-level libraries like libc.
225 Allow_undefined_symbols bool
226
227 // don't link in crt_begin and crt_end. This flag should only be necessary for
228 // compiling crt or libc.
229 Nocrt bool `android:"arch_variant"`
230
231 // don't insert default compiler flags into asflags, cflags,
232 // cppflags, conlyflags, ldflags, or include_dirs
233 No_default_compiler_flags bool
234
235 // compile module with clang instead of gcc
236 Clang bool `android:"arch_variant"`
237
238 // pass -frtti instead of -fno-rtti
239 Rtti bool
240
241 // -l arguments to pass to linker for host-provided shared libraries
242 Host_ldlibs []string `android:"arch_variant"`
243
244 // select the STL library to use. Possible values are "libc++", "libc++_static",
245 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
246 // default
247 Stl string
248
249 // Set for combined shared/static libraries to prevent compiling object files a second time
250 SkipCompileObjs bool `blueprint:"mutated"`
251
252 Debug, Release struct {
253 // list of module-specific flags that will be used for C and C++ compiles in debug or
254 // release builds
255 Cflags []string `android:"arch_variant"`
256 } `android:"arch_variant"`
257
258 // Minimum sdk version supported when compiling against the ndk
259 Sdk_version string
260
261 // install to a subdirectory of the default install path for the module
262 Relative_install_path string
263}
264
Colin Crossfa138792015-04-24 17:31:52 -0700265// CCBase contains the properties and members used by all C/C++ module types, and implements
Colin Crossc472d572015-03-17 15:06:21 -0700266// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
267// and uses a ccModuleType interface to that struct to create the build steps.
Colin Crossfa138792015-04-24 17:31:52 -0700268type CCBase struct {
Colin Crossc472d572015-03-17 15:06:21 -0700269 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700270 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700271
Colin Cross7d5136f2015-05-11 13:39:40 -0700272 Properties CCBaseProperties
Colin Crossfa138792015-04-24 17:31:52 -0700273
274 unused struct {
275 Asan bool
276 Native_coverage bool
277 Strip string
278 Tags []string
279 Required []string
280 }
Colin Crossc472d572015-03-17 15:06:21 -0700281
282 installPath string
Colin Cross74d1ec02015-04-28 13:30:13 -0700283
284 savedDepNames CCDeps
Colin Crossc472d572015-03-17 15:06:21 -0700285}
286
Colin Crossfa138792015-04-24 17:31:52 -0700287func newCCBase(base *CCBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700288 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
289
290 base.module = module
291
Colin Crossfa138792015-04-24 17:31:52 -0700292 props = append(props, &base.Properties, &base.unused)
Colin Crossc472d572015-03-17 15:06:21 -0700293
Colin Cross5049f022015-03-18 13:28:46 -0700294 return common.InitAndroidArchModule(module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700295}
296
Colin Crossfa138792015-04-24 17:31:52 -0700297func (c *CCBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800298 toolchain := c.findToolchain(ctx)
299 if ctx.Failed() {
300 return
301 }
302
Colin Cross21b9a242015-03-24 14:15:58 -0700303 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800304 if ctx.Failed() {
305 return
306 }
307
Colin Cross74d1ec02015-04-28 13:30:13 -0700308 deps := c.depsToPaths(ctx, c.savedDepNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800309 if ctx.Failed() {
310 return
311 }
312
Colin Cross28344522015-04-22 13:07:53 -0700313 flags.CFlags = append(flags.CFlags, deps.Cflags...)
Colin Crossed9f8682015-03-18 17:17:35 -0700314
Colin Cross581c1892015-04-07 16:50:10 -0700315 objFiles := c.compileObjs(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800316 if ctx.Failed() {
317 return
318 }
319
Colin Cross581c1892015-04-07 16:50:10 -0700320 generatedObjFiles := c.compileGeneratedObjs(ctx, flags)
Colin Cross5049f022015-03-18 13:28:46 -0700321 if ctx.Failed() {
322 return
323 }
324
325 objFiles = append(objFiles, generatedObjFiles...)
326
Colin Cross3f40fa42015-01-30 17:27:36 -0800327 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
328 if ctx.Failed() {
329 return
330 }
Dan Albertc403f7c2015-03-18 14:01:18 -0700331
332 c.ccModuleType().installModule(ctx, flags)
333 if ctx.Failed() {
334 return
335 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800336}
337
Colin Crossfa138792015-04-24 17:31:52 -0700338func (c *CCBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800339 return c.module
340}
341
Colin Crossfa138792015-04-24 17:31:52 -0700342var _ common.AndroidDynamicDepender = (*CCBase)(nil)
Colin Cross3f40fa42015-01-30 17:27:36 -0800343
Colin Crossfa138792015-04-24 17:31:52 -0700344func (c *CCBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800345 arch := ctx.Arch()
Colin Crossd3ba0392015-05-07 14:11:29 -0700346 hod := ctx.HostOrDevice()
347 factory := toolchainFactories[hod][arch.ArchType]
Colin Cross3f40fa42015-01-30 17:27:36 -0800348 if factory == nil {
349 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
Colin Crossd3ba0392015-05-07 14:11:29 -0700350 hod.String(), arch.String()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800351 }
352 return factory(arch.ArchVariant, arch.CpuVariant)
353}
354
Colin Crossfa138792015-04-24 17:31:52 -0700355func (c *CCBase) ModifyProperties(ctx common.AndroidBaseContext) {
356}
357
Colin Crosse11befc2015-04-27 17:49:17 -0700358func (c *CCBase) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossfa138792015-04-24 17:31:52 -0700359 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.Properties.Whole_static_libs...)
360 depNames.StaticLibs = append(depNames.StaticLibs, c.Properties.Static_libs...)
361 depNames.SharedLibs = append(depNames.SharedLibs, c.Properties.Shared_libs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700362
Colin Cross21b9a242015-03-24 14:15:58 -0700363 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800364}
365
Colin Crossfa138792015-04-24 17:31:52 -0700366func (c *CCBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
367 c.module.ModifyProperties(ctx)
368
Colin Cross74d1ec02015-04-28 13:30:13 -0700369 c.savedDepNames = c.module.depNames(ctx, CCDeps{})
370 c.savedDepNames.WholeStaticLibs = lastUniqueElements(c.savedDepNames.WholeStaticLibs)
371 c.savedDepNames.StaticLibs = lastUniqueElements(c.savedDepNames.StaticLibs)
372 c.savedDepNames.SharedLibs = lastUniqueElements(c.savedDepNames.SharedLibs)
373
374 staticLibs := c.savedDepNames.WholeStaticLibs
375 staticLibs = append(staticLibs, c.savedDepNames.StaticLibs...)
376 staticLibs = append(staticLibs, c.savedDepNames.LateStaticLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700377 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800378
Colin Cross74d1ec02015-04-28 13:30:13 -0700379 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, c.savedDepNames.SharedLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700380
Colin Cross74d1ec02015-04-28 13:30:13 -0700381 ret := append([]string(nil), c.savedDepNames.ObjFiles...)
382 if c.savedDepNames.CrtBegin != "" {
383 ret = append(ret, c.savedDepNames.CrtBegin)
Colin Cross21b9a242015-03-24 14:15:58 -0700384 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700385 if c.savedDepNames.CrtEnd != "" {
386 ret = append(ret, c.savedDepNames.CrtEnd)
Colin Cross21b9a242015-03-24 14:15:58 -0700387 }
388
389 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800390}
391
392// Create a ccFlags struct that collects the compile flags from global values,
393// per-target values, module type values, and per-module Blueprints properties
Colin Crossfa138792015-04-24 17:31:52 -0700394func (c *CCBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700395 flags := CCFlags{
Colin Crossfa138792015-04-24 17:31:52 -0700396 CFlags: c.Properties.Cflags,
397 CppFlags: c.Properties.Cppflags,
398 ConlyFlags: c.Properties.Conlyflags,
399 LdFlags: c.Properties.Ldflags,
400 AsFlags: c.Properties.Asflags,
401 YaccFlags: c.Properties.Yaccflags,
402 Nocrt: c.Properties.Nocrt,
Colin Cross97ba0732015-03-23 17:50:24 -0700403 Toolchain: toolchain,
Colin Crossfa138792015-04-24 17:31:52 -0700404 Clang: c.Properties.Clang,
Colin Cross3f40fa42015-01-30 17:27:36 -0800405 }
Colin Cross28344522015-04-22 13:07:53 -0700406
407 // Include dir cflags
Colin Crossf2298272015-05-12 11:36:53 -0700408 common.CheckSrcDirsExist(ctx, c.Properties.Include_dirs, "include_dirs")
409 common.CheckModuleSrcDirsExist(ctx, c.Properties.Local_include_dirs, "local_include_dirs")
410
Colin Crossfa138792015-04-24 17:31:52 -0700411 rootIncludeDirs := pathtools.PrefixPaths(c.Properties.Include_dirs, ctx.AConfig().SrcDir())
412 localIncludeDirs := pathtools.PrefixPaths(c.Properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -0700413 flags.GlobalFlags = append(flags.GlobalFlags,
414 includeDirsToFlags(rootIncludeDirs),
415 includeDirsToFlags(localIncludeDirs))
416
Colin Crossfa138792015-04-24 17:31:52 -0700417 if !c.Properties.No_default_compiler_flags {
418 if c.Properties.Sdk_version == "" || ctx.Host() {
Colin Cross28344522015-04-22 13:07:53 -0700419 flags.GlobalFlags = append(flags.GlobalFlags,
420 "${commonGlobalIncludes}",
421 toolchain.IncludeFlags(),
422 "-I${SrcDir}/libnativehelper/include/nativehelper")
423 }
424
425 flags.GlobalFlags = append(flags.GlobalFlags, []string{
426 "-I" + common.ModuleSrcDir(ctx),
427 "-I" + common.ModuleOutDir(ctx),
428 "-I" + common.ModuleGenDir(ctx),
429 }...)
430 }
431
Colin Crossfa138792015-04-24 17:31:52 -0700432 instructionSet := c.Properties.Instruction_set
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700433 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
434 if err != nil {
435 ctx.ModuleErrorf("%s", err)
436 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800437
Colin Crossaf19a292015-03-18 12:07:10 -0700438 // TODO: debug
Colin Crossfa138792015-04-24 17:31:52 -0700439 flags.CFlags = append(flags.CFlags, c.Properties.Release.Cflags...)
Colin Crossaf19a292015-03-18 12:07:10 -0700440
Colin Cross28d76592015-03-26 16:14:04 -0700441 if ctx.Host() && !ctx.ContainsProperty("clang") {
Colin Cross97ba0732015-03-23 17:50:24 -0700442 flags.Clang = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800443 }
444
Colin Cross97ba0732015-03-23 17:50:24 -0700445 if flags.Clang {
446 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
Colin Crossfa138792015-04-24 17:31:52 -0700447 flags.CFlags = append(flags.CFlags, c.Properties.Clang_cflags...)
448 flags.AsFlags = append(flags.AsFlags, c.Properties.Clang_asflags...)
Colin Cross97ba0732015-03-23 17:50:24 -0700449 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
450 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
451 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800452
Colin Cross97ba0732015-03-23 17:50:24 -0700453 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
454 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700455 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700456 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700457 }
458
Colin Cross3f40fa42015-01-30 17:27:36 -0800459 target := "-target " + toolchain.ClangTriple()
460 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
461
Colin Cross97ba0732015-03-23 17:50:24 -0700462 flags.CFlags = append(flags.CFlags, target, gccPrefix)
463 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
464 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800465 }
466
Colin Crossfa138792015-04-24 17:31:52 -0700467 if !c.Properties.No_default_compiler_flags {
468 if ctx.Device() && !c.Properties.Allow_undefined_symbols {
Colin Cross97ba0732015-03-23 17:50:24 -0700469 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800470 }
471
Colin Cross56b4d452015-04-21 17:38:44 -0700472 flags.GlobalFlags = append(flags.GlobalFlags, instructionSetFlags)
473
Colin Cross97ba0732015-03-23 17:50:24 -0700474 if flags.Clang {
475 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700476 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800477 toolchain.ClangCflags(),
478 "${commonClangGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700479 fmt.Sprintf("${%sClangGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800480 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700481 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700482 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800483 toolchain.Cflags(),
484 "${commonGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700485 fmt.Sprintf("${%sGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800486 }
487
Colin Crossf6566ed2015-03-24 11:13:38 -0700488 if ctx.Device() {
Colin Crossfa138792015-04-24 17:31:52 -0700489 if c.Properties.Rtti {
Colin Cross97ba0732015-03-23 17:50:24 -0700490 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800491 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700492 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800493 }
494 }
495
Colin Cross97ba0732015-03-23 17:50:24 -0700496 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800497
Colin Cross97ba0732015-03-23 17:50:24 -0700498 if flags.Clang {
499 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
500 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800501 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700502 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
503 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800504 }
Colin Cross28344522015-04-22 13:07:53 -0700505
506 if ctx.Host() {
Colin Crossfa138792015-04-24 17:31:52 -0700507 flags.LdFlags = append(flags.LdFlags, c.Properties.Host_ldlibs...)
Colin Cross28344522015-04-22 13:07:53 -0700508 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800509 }
510
Colin Cross0676e2d2015-04-24 17:39:18 -0700511 flags = c.ccModuleType().flags(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800512
513 // Optimization to reduce size of build.ninja
514 // Replace the long list of flags for each file with a module-local variable
Colin Cross97ba0732015-03-23 17:50:24 -0700515 ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " "))
516 ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " "))
517 ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " "))
518 flags.CFlags = []string{"$cflags"}
519 flags.CppFlags = []string{"$cppflags"}
520 flags.AsFlags = []string{"$asflags"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800521
522 return flags
523}
524
Colin Cross0676e2d2015-04-24 17:39:18 -0700525func (c *CCBase) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross3f40fa42015-01-30 17:27:36 -0800526 return flags
527}
528
529// Compile a list of source files into objects a specified subdirectory
Colin Crossfa138792015-04-24 17:31:52 -0700530func (c *CCBase) customCompileObjs(ctx common.AndroidModuleContext, flags CCFlags,
Colin Cross581c1892015-04-07 16:50:10 -0700531 subdir string, srcFiles []string) []string {
532
533 buildFlags := ccFlagsToBuilderFlags(flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800534
Colin Cross8f101b42015-06-17 15:09:06 -0700535 srcFiles = ctx.ExpandSources(srcFiles)
Colin Cross581c1892015-04-07 16:50:10 -0700536 srcFiles, deps := genSources(ctx, srcFiles, buildFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800537
Colin Cross581c1892015-04-07 16:50:10 -0700538 return TransformSourceToObj(ctx, subdir, srcFiles, buildFlags, deps)
Colin Cross3f40fa42015-01-30 17:27:36 -0800539}
540
Colin Crossfa138792015-04-24 17:31:52 -0700541// Compile files listed in c.Properties.Srcs into objects
542func (c *CCBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800543
Colin Crossfa138792015-04-24 17:31:52 -0700544 if c.Properties.SkipCompileObjs {
Colin Cross3f40fa42015-01-30 17:27:36 -0800545 return nil
546 }
547
Colin Crossfa138792015-04-24 17:31:52 -0700548 return c.customCompileObjs(ctx, flags, "", c.Properties.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800549}
550
Colin Cross5049f022015-03-18 13:28:46 -0700551// Compile generated source files from dependencies
Colin Crossfa138792015-04-24 17:31:52 -0700552func (c *CCBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700553 var srcs []string
554
Colin Crossfa138792015-04-24 17:31:52 -0700555 if c.Properties.SkipCompileObjs {
Colin Cross5049f022015-03-18 13:28:46 -0700556 return nil
557 }
558
559 ctx.VisitDirectDeps(func(module blueprint.Module) {
560 if gen, ok := module.(genrule.SourceFileGenerator); ok {
561 srcs = append(srcs, gen.GeneratedSourceFiles()...)
562 }
563 })
564
565 if len(srcs) == 0 {
566 return nil
567 }
568
Colin Cross581c1892015-04-07 16:50:10 -0700569 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags), nil)
Colin Cross5049f022015-03-18 13:28:46 -0700570}
571
Colin Crossfa138792015-04-24 17:31:52 -0700572func (c *CCBase) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800573 return ""
574}
575
Colin Crossfa138792015-04-24 17:31:52 -0700576func (c *CCBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800577 names []string) (modules []common.AndroidModule,
Colin Cross28344522015-04-22 13:07:53 -0700578 outputFiles []string, exportedFlags []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800579
580 for _, n := range names {
581 found := false
582 ctx.VisitDirectDeps(func(m blueprint.Module) {
583 otherName := ctx.OtherModuleName(m)
584 if otherName != n {
585 return
586 }
587
Colin Cross97ba0732015-03-23 17:50:24 -0700588 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800589 if a.Disabled() {
590 // If a cc_library host+device module depends on a library that exists as both
591 // cc_library_shared and cc_library_host_shared, it will end up with two
592 // dependencies with the same name, one of which is marked disabled for each
593 // of host and device. Ignore the disabled one.
594 return
595 }
Colin Crossd3ba0392015-05-07 14:11:29 -0700596 if a.HostOrDevice() != ctx.HostOrDevice() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800597 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
598 otherName)
599 return
600 }
601
602 if outputFile := a.outputFile(); outputFile != "" {
603 if found {
604 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
605 return
606 }
607 outputFiles = append(outputFiles, outputFile)
608 modules = append(modules, a)
Colin Cross28344522015-04-22 13:07:53 -0700609 if i, ok := a.(ccExportedFlagsProducer); ok {
610 exportedFlags = append(exportedFlags, i.exportedFlags()...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800611 }
612 found = true
613 } else {
614 ctx.ModuleErrorf("module %q missing output file", otherName)
615 return
616 }
617 } else {
618 ctx.ModuleErrorf("module %q not an android module", otherName)
619 return
620 }
621 })
622 if !found {
623 ctx.ModuleErrorf("unsatisified dependency on %q", n)
624 }
625 }
626
Colin Cross28344522015-04-22 13:07:53 -0700627 return modules, outputFiles, exportedFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800628}
629
Colin Cross21b9a242015-03-24 14:15:58 -0700630// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
631// containing paths
Colin Crossfa138792015-04-24 17:31:52 -0700632func (c *CCBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700633 var depPaths CCDeps
Colin Cross28344522015-04-22 13:07:53 -0700634 var newCflags []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800635
Colin Cross21b9a242015-03-24 14:15:58 -0700636 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800637
Colin Cross28344522015-04-22 13:07:53 -0700638 wholeStaticLibModules, depPaths.WholeStaticLibs, newCflags =
Colin Cross21b9a242015-03-24 14:15:58 -0700639 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
Colin Cross28344522015-04-22 13:07:53 -0700640 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800641
Colin Cross21b9a242015-03-24 14:15:58 -0700642 for _, m := range wholeStaticLibModules {
643 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
644 depPaths.WholeStaticLibObjFiles =
645 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
646 } else {
647 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
648 }
649 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800650
Colin Cross28344522015-04-22 13:07:53 -0700651 _, depPaths.StaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.StaticLibs)
652 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700653
Colin Cross28344522015-04-22 13:07:53 -0700654 _, depPaths.LateStaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
655 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700656
Colin Cross28344522015-04-22 13:07:53 -0700657 _, depPaths.SharedLibs, newCflags = c.depsToPathsFromList(ctx, depNames.SharedLibs)
658 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700659
660 ctx.VisitDirectDeps(func(m blueprint.Module) {
Dan Albertc3144b12015-04-28 18:17:56 -0700661 if obj, ok := m.(ccObjectProvider); ok {
Colin Cross21b9a242015-03-24 14:15:58 -0700662 otherName := ctx.OtherModuleName(m)
663 if otherName == depNames.CrtBegin {
Colin Crossfa138792015-04-24 17:31:52 -0700664 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700665 depPaths.CrtBegin = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700666 }
667 } else if otherName == depNames.CrtEnd {
Colin Crossfa138792015-04-24 17:31:52 -0700668 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700669 depPaths.CrtEnd = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700670 }
671 } else {
Dan Albertc3144b12015-04-28 18:17:56 -0700672 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.object().outputFile())
Colin Cross21b9a242015-03-24 14:15:58 -0700673 }
674 }
675 })
676
677 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800678}
679
Colin Cross7d5136f2015-05-11 13:39:40 -0700680type ccLinkedProperties struct {
681 VariantIsShared bool `blueprint:"mutated"`
682 VariantIsStatic bool `blueprint:"mutated"`
683 VariantIsStaticBinary bool `blueprint:"mutated"`
684}
685
Colin Crossfa138792015-04-24 17:31:52 -0700686// CCLinked contains the properties and members used by libraries and executables
687type CCLinked struct {
688 CCBase
Colin Cross7d5136f2015-05-11 13:39:40 -0700689 dynamicProperties ccLinkedProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800690}
691
Colin Crossfa138792015-04-24 17:31:52 -0700692func newCCDynamic(dynamic *CCLinked, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700693 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
694
Colin Crossed4cf0b2015-03-26 14:43:45 -0700695 props = append(props, &dynamic.dynamicProperties)
696
Colin Crossfa138792015-04-24 17:31:52 -0700697 return newCCBase(&dynamic.CCBase, module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700698}
699
Colin Crossfa138792015-04-24 17:31:52 -0700700func (c *CCLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross28d76592015-03-26 16:14:04 -0700701 if ctx.ContainsProperty("system_shared_libs") {
Colin Crossfa138792015-04-24 17:31:52 -0700702 return c.Properties.System_shared_libs
703 } else if ctx.Device() && c.Properties.Sdk_version == "" {
Colin Cross577f6e42015-03-27 18:23:34 -0700704 return []string{"libc", "libm"}
Colin Cross28d76592015-03-26 16:14:04 -0700705 } else {
Colin Cross577f6e42015-03-27 18:23:34 -0700706 return nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800707 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800708}
709
Colin Crossfa138792015-04-24 17:31:52 -0700710func (c *CCLinked) stl(ctx common.AndroidBaseContext) string {
711 if c.Properties.Sdk_version != "" && ctx.Device() {
712 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700713 case "":
714 return "ndk_system"
715 case "c++_shared", "c++_static",
716 "stlport_shared", "stlport_static",
717 "gnustl_static":
Colin Crossfa138792015-04-24 17:31:52 -0700718 return "ndk_lib" + c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700719 default:
Colin Crossfa138792015-04-24 17:31:52 -0700720 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700721 return ""
722 }
723 }
724
Colin Crossfa138792015-04-24 17:31:52 -0700725 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700726 case "libc++", "libc++_static",
727 "stlport", "stlport_static",
728 "libstdc++":
Colin Crossfa138792015-04-24 17:31:52 -0700729 return c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700730 case "none":
731 return ""
732 case "":
Colin Cross18b6dc52015-04-28 13:20:37 -0700733 if c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700734 return "libc++_static"
Colin Cross18b6dc52015-04-28 13:20:37 -0700735 } else {
736 return "libc++" // TODO: mingw needs libstdc++
Colin Crossed4cf0b2015-03-26 14:43:45 -0700737 }
738 default:
Colin Crossfa138792015-04-24 17:31:52 -0700739 ctx.ModuleErrorf("stl: %q is not a supported STL", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700740 return ""
741 }
742}
743
Colin Cross0af4b842015-04-30 16:36:18 -0700744var hostDynamicGccLibs, hostStaticGccLibs []string
745
746func init() {
747 if runtime.GOOS == "darwin" {
748 hostDynamicGccLibs = []string{"-lc", "-lSystem"}
749 hostStaticGccLibs = []string{"NO_STATIC_HOST_BINARIES_ON_DARWIN"}
750 } else {
751 hostDynamicGccLibs = []string{"-lgcc_s", "-lgcc", "-lc", "-lgcc_s", "-lgcc"}
752 hostStaticGccLibs = []string{"-Wl,--start-group", "-lgcc", "-lgcc_eh", "-lc", "-Wl,--end-group"}
753 }
754}
Colin Cross712fc022015-04-27 11:13:34 -0700755
Colin Crosse11befc2015-04-27 17:49:17 -0700756func (c *CCLinked) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700757 stl := c.stl(ctx)
758 if ctx.Failed() {
759 return flags
760 }
761
762 switch stl {
763 case "libc++", "libc++_static":
764 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
Colin Cross28344522015-04-22 13:07:53 -0700765 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/external/libcxx/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700766 if ctx.Host() {
767 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
768 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross712fc022015-04-27 11:13:34 -0700769 flags.LdFlags = append(flags.LdFlags, "-lm", "-lpthread")
Colin Cross18b6dc52015-04-28 13:20:37 -0700770 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700771 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700772 } else {
773 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700774 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700775 }
776 case "stlport", "stlport_static":
777 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700778 flags.CFlags = append(flags.CFlags,
779 "-I${SrcDir}/external/stlport/stlport",
780 "-I${SrcDir}/bionic/libstdc++/include",
781 "-I${SrcDir}/bionic")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700782 }
783 case "libstdc++":
784 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
785 // tree is in good enough shape to not need it.
786 // Host builds will use GNU libstdc++.
787 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700788 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/bionic/libstdc++/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700789 }
790 case "ndk_system":
Colin Cross1332b002015-04-07 17:11:30 -0700791 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources/"
Colin Cross28344522015-04-22 13:07:53 -0700792 flags.CFlags = append(flags.CFlags, "-isystem "+ndkSrcRoot+"cxx-stl/system/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700793 case "ndk_libc++_shared", "ndk_libc++_static":
794 // TODO(danalbert): This really shouldn't be here...
795 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
796 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
797 // Nothing
798 case "":
799 // None or error.
800 if ctx.Host() {
801 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
802 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross18b6dc52015-04-28 13:20:37 -0700803 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700804 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700805 } else {
806 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700807 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700808 }
809 default:
Colin Crossfa138792015-04-24 17:31:52 -0700810 panic(fmt.Errorf("Unknown stl in CCLinked.Flags: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700811 }
812
813 return flags
814}
815
Colin Crosse11befc2015-04-27 17:49:17 -0700816func (c *CCLinked) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
817 depNames = c.CCBase.depNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800818
Colin Crossed4cf0b2015-03-26 14:43:45 -0700819 stl := c.stl(ctx)
820 if ctx.Failed() {
821 return depNames
822 }
823
824 switch stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700825 case "libstdc++":
826 if ctx.Device() {
827 depNames.SharedLibs = append(depNames.SharedLibs, stl)
828 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700829 case "libc++", "libc++_static":
830 if stl == "libc++" {
831 depNames.SharedLibs = append(depNames.SharedLibs, stl)
832 } else {
833 depNames.StaticLibs = append(depNames.StaticLibs, stl)
834 }
835 if ctx.Device() {
836 if ctx.Arch().ArchType == common.Arm {
837 depNames.StaticLibs = append(depNames.StaticLibs, "libunwind_llvm")
838 }
839 if c.staticBinary() {
840 depNames.StaticLibs = append(depNames.StaticLibs, "libdl")
841 } else {
842 depNames.SharedLibs = append(depNames.SharedLibs, "libdl")
843 }
844 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700845 case "stlport":
846 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
847 case "stlport_static":
848 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
849 case "":
850 // None or error.
851 case "ndk_system":
852 // TODO: Make a system STL prebuilt for the NDK.
853 // 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 -0700854 // its own includes. The includes are handled in CCBase.Flags().
Colin Cross577f6e42015-03-27 18:23:34 -0700855 depNames.SharedLibs = append([]string{"libstdc++"}, depNames.SharedLibs...)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700856 case "ndk_libc++_shared", "ndk_libstlport_shared":
857 depNames.SharedLibs = append(depNames.SharedLibs, stl)
858 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
859 depNames.StaticLibs = append(depNames.StaticLibs, stl)
860 default:
Colin Crosse11befc2015-04-27 17:49:17 -0700861 panic(fmt.Errorf("Unknown stl in CCLinked.depNames: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700862 }
863
Colin Cross74d1ec02015-04-28 13:30:13 -0700864 if ctx.ModuleName() != "libcompiler_rt-extras" {
865 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
866 }
867
Colin Crossf6566ed2015-03-24 11:13:38 -0700868 if ctx.Device() {
Colin Cross77b00fa2015-03-16 16:15:49 -0700869 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700870 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700871
Colin Cross18b6dc52015-04-28 13:20:37 -0700872 if !c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700873 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
874 }
Colin Cross577f6e42015-03-27 18:23:34 -0700875
Colin Crossfa138792015-04-24 17:31:52 -0700876 if c.Properties.Sdk_version != "" {
877 version := c.Properties.Sdk_version
Colin Cross577f6e42015-03-27 18:23:34 -0700878 depNames.SharedLibs = append(depNames.SharedLibs,
879 "ndk_libc."+version,
880 "ndk_libm."+version,
881 )
882 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800883 }
884
Colin Cross21b9a242015-03-24 14:15:58 -0700885 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800886}
887
Colin Crossed4cf0b2015-03-26 14:43:45 -0700888// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
889type ccLinkedInterface interface {
890 // Returns true if the build options for the module have selected a static or shared build
891 buildStatic() bool
892 buildShared() bool
893
894 // Sets whether a specific variant is static or shared
Colin Cross18b6dc52015-04-28 13:20:37 -0700895 setStatic(bool)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700896
Colin Cross18b6dc52015-04-28 13:20:37 -0700897 // Returns whether a specific variant is a static library or binary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700898 static() bool
Colin Cross18b6dc52015-04-28 13:20:37 -0700899
900 // Returns whether a module is a static binary
901 staticBinary() bool
Colin Crossed4cf0b2015-03-26 14:43:45 -0700902}
903
904var _ ccLinkedInterface = (*CCLibrary)(nil)
905var _ ccLinkedInterface = (*CCBinary)(nil)
906
Colin Crossfa138792015-04-24 17:31:52 -0700907func (c *CCLinked) static() bool {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700908 return c.dynamicProperties.VariantIsStatic
909}
910
Colin Cross18b6dc52015-04-28 13:20:37 -0700911func (c *CCLinked) staticBinary() bool {
912 return c.dynamicProperties.VariantIsStaticBinary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700913}
914
Colin Cross18b6dc52015-04-28 13:20:37 -0700915func (c *CCLinked) setStatic(static bool) {
916 c.dynamicProperties.VariantIsStatic = static
Colin Crossed4cf0b2015-03-26 14:43:45 -0700917}
918
Colin Cross28344522015-04-22 13:07:53 -0700919type ccExportedFlagsProducer interface {
920 exportedFlags() []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800921}
922
923//
924// Combined static+shared libraries
925//
926
Colin Cross7d5136f2015-05-11 13:39:40 -0700927type CCLibraryProperties struct {
928 BuildStatic bool `blueprint:"mutated"`
929 BuildShared bool `blueprint:"mutated"`
930 Static struct {
931 Srcs []string `android:"arch_variant"`
932 Cflags []string `android:"arch_variant"`
933 Whole_static_libs []string `android:"arch_variant"`
934 Static_libs []string `android:"arch_variant"`
935 Shared_libs []string `android:"arch_variant"`
936 } `android:"arch_variant"`
937 Shared struct {
938 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"`
943 } `android:"arch_variant"`
944}
945
Colin Cross97ba0732015-03-23 17:50:24 -0700946type CCLibrary struct {
Colin Crossfa138792015-04-24 17:31:52 -0700947 CCLinked
Colin Cross3f40fa42015-01-30 17:27:36 -0800948
Colin Cross28344522015-04-22 13:07:53 -0700949 reuseFrom ccLibraryInterface
950 reuseObjFiles []string
951 objFiles []string
952 exportFlags []string
953 out string
Colin Cross3f40fa42015-01-30 17:27:36 -0800954
Colin Cross7d5136f2015-05-11 13:39:40 -0700955 LibraryProperties CCLibraryProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800956}
957
Colin Crossed4cf0b2015-03-26 14:43:45 -0700958func (c *CCLibrary) buildStatic() bool {
959 return c.LibraryProperties.BuildStatic
960}
961
962func (c *CCLibrary) buildShared() bool {
963 return c.LibraryProperties.BuildShared
964}
965
Colin Cross97ba0732015-03-23 17:50:24 -0700966type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700967 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -0700968 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700969 setReuseFrom(ccLibraryInterface)
970 getReuseFrom() ccLibraryInterface
971 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -0700972 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -0700973}
974
Colin Crossed4cf0b2015-03-26 14:43:45 -0700975var _ ccLibraryInterface = (*CCLibrary)(nil)
976
Colin Cross97ba0732015-03-23 17:50:24 -0700977func (c *CCLibrary) ccLibrary() *CCLibrary {
978 return c
Colin Cross3f40fa42015-01-30 17:27:36 -0800979}
980
Colin Cross97ba0732015-03-23 17:50:24 -0700981func NewCCLibrary(library *CCLibrary, module CCModuleType,
982 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
983
Colin Crossfa138792015-04-24 17:31:52 -0700984 return newCCDynamic(&library.CCLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -0700985 &library.LibraryProperties)
986}
987
988func CCLibraryFactory() (blueprint.Module, []interface{}) {
989 module := &CCLibrary{}
990
991 module.LibraryProperties.BuildShared = true
992 module.LibraryProperties.BuildStatic = true
993
994 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
995}
996
Colin Cross0676e2d2015-04-24 17:39:18 -0700997func (c *CCLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -0700998 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Cross2732e9a2015-04-28 13:23:52 -0700999 if c.static() {
1000 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Static.Whole_static_libs...)
1001 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Static.Static_libs...)
1002 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Static.Shared_libs...)
1003 } else {
Colin Crossf6566ed2015-03-24 11:13:38 -07001004 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001005 if c.Properties.Sdk_version == "" {
1006 depNames.CrtBegin = "crtbegin_so"
1007 depNames.CrtEnd = "crtend_so"
1008 } else {
1009 depNames.CrtBegin = "ndk_crtbegin_so." + c.Properties.Sdk_version
1010 depNames.CrtEnd = "ndk_crtend_so." + c.Properties.Sdk_version
1011 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001012 }
Colin Cross2732e9a2015-04-28 13:23:52 -07001013 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Shared.Whole_static_libs...)
1014 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Shared.Static_libs...)
1015 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Shared.Shared_libs...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001016 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001017
Colin Cross21b9a242015-03-24 14:15:58 -07001018 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001019}
1020
Colin Cross97ba0732015-03-23 17:50:24 -07001021func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001022 return c.out
1023}
1024
Colin Crossed4cf0b2015-03-26 14:43:45 -07001025func (c *CCLibrary) getReuseObjFiles() []string {
1026 return c.reuseObjFiles
1027}
1028
1029func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
1030 c.reuseFrom = reuseFrom
1031}
1032
1033func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
1034 return c.reuseFrom
1035}
1036
Colin Cross97ba0732015-03-23 17:50:24 -07001037func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001038 return c.objFiles
1039}
1040
Colin Cross28344522015-04-22 13:07:53 -07001041func (c *CCLibrary) exportedFlags() []string {
1042 return c.exportFlags
Colin Cross3f40fa42015-01-30 17:27:36 -08001043}
1044
Colin Cross0676e2d2015-04-24 17:39:18 -07001045func (c *CCLibrary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001046 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001047
Colin Cross97ba0732015-03-23 17:50:24 -07001048 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -08001049
Colin Crossd8e780d2015-04-28 17:39:43 -07001050 if c.static() {
1051 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Static.Cflags...)
1052 } else {
1053 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Shared.Cflags...)
1054 }
1055
Colin Cross18b6dc52015-04-28 13:20:37 -07001056 if !c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001057 libName := ctx.ModuleName()
1058 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
1059 sharedFlag := "-Wl,-shared"
Colin Crossfa138792015-04-24 17:31:52 -07001060 if c.Properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001061 sharedFlag = "-shared"
1062 }
Colin Crossf6566ed2015-03-24 11:13:38 -07001063 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001064 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001065 }
Colin Cross97ba0732015-03-23 17:50:24 -07001066
Colin Cross0af4b842015-04-30 16:36:18 -07001067 if ctx.Darwin() {
1068 flags.LdFlags = append(flags.LdFlags,
1069 "-dynamiclib",
1070 "-single_module",
1071 //"-read_only_relocs suppress",
1072 "-install_name @rpath/"+libName+sharedLibraryExtension,
1073 )
1074 } else {
1075 flags.LdFlags = append(flags.LdFlags,
1076 "-Wl,--gc-sections",
1077 sharedFlag,
1078 "-Wl,-soname,"+libName+sharedLibraryExtension,
1079 )
1080 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001081 }
Colin Cross97ba0732015-03-23 17:50:24 -07001082
1083 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001084}
1085
Colin Cross97ba0732015-03-23 17:50:24 -07001086func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1087 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001088
1089 staticFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001090 objFilesStatic := c.customCompileObjs(ctx, staticFlags, common.DeviceStaticLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001091 c.LibraryProperties.Static.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001092
1093 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001094 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001095
1096 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1097
Colin Cross0af4b842015-04-30 16:36:18 -07001098 if ctx.Darwin() {
1099 TransformDarwinObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1100 } else {
1101 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1102 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001103
1104 c.objFiles = objFiles
1105 c.out = outputFile
Colin Crossf2298272015-05-12 11:36:53 -07001106
1107 common.CheckModuleSrcDirsExist(ctx, c.Properties.Export_include_dirs, "export_include_dirs")
Colin Crossfa138792015-04-24 17:31:52 -07001108 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001109 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001110
1111 ctx.CheckbuildFile(outputFile)
1112}
1113
Colin Cross97ba0732015-03-23 17:50:24 -07001114func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1115 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001116
1117 sharedFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001118 objFilesShared := c.customCompileObjs(ctx, sharedFlags, common.DeviceSharedLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -07001119 c.LibraryProperties.Shared.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001120
1121 objFiles = append(objFiles, objFilesShared...)
1122
1123 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1124
Colin Cross97ba0732015-03-23 17:50:24 -07001125 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001126 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, false,
Colin Cross77b00fa2015-03-16 16:15:49 -07001127 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001128
1129 c.out = outputFile
Colin Crossfa138792015-04-24 17:31:52 -07001130 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001131 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001132}
1133
Colin Cross97ba0732015-03-23 17:50:24 -07001134func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1135 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001136
1137 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001138 if c.getReuseFrom().ccLibrary() == c {
1139 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001140 } else {
Colin Cross2732e9a2015-04-28 13:23:52 -07001141 if c.getReuseFrom().ccLibrary().LibraryProperties.Static.Cflags == nil &&
1142 c.LibraryProperties.Shared.Cflags == nil {
1143 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
1144 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001145 }
1146
Colin Crossed4cf0b2015-03-26 14:43:45 -07001147 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001148 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1149 } else {
1150 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1151 }
1152}
1153
Colin Cross97ba0732015-03-23 17:50:24 -07001154func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001155 // Static libraries do not get installed.
1156}
1157
Colin Cross97ba0732015-03-23 17:50:24 -07001158func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001159 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001160 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001161 installDir = "lib64"
1162 }
1163
Colin Crossfa138792015-04-24 17:31:52 -07001164 ctx.InstallFile(filepath.Join(installDir, c.Properties.Relative_install_path), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001165}
1166
Colin Cross97ba0732015-03-23 17:50:24 -07001167func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001168 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001169 c.installStaticLibrary(ctx, flags)
1170 } else {
1171 c.installSharedLibrary(ctx, flags)
1172 }
1173}
1174
Colin Cross3f40fa42015-01-30 17:27:36 -08001175//
1176// Objects (for crt*.o)
1177//
1178
Dan Albertc3144b12015-04-28 18:17:56 -07001179type ccObjectProvider interface {
1180 object() *ccObject
1181}
1182
Colin Cross3f40fa42015-01-30 17:27:36 -08001183type ccObject struct {
Colin Crossfa138792015-04-24 17:31:52 -07001184 CCBase
Colin Cross3f40fa42015-01-30 17:27:36 -08001185 out string
1186}
1187
Dan Albertc3144b12015-04-28 18:17:56 -07001188func (c *ccObject) object() *ccObject {
1189 return c
1190}
1191
Colin Cross97ba0732015-03-23 17:50:24 -07001192func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001193 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001194
Colin Crossfa138792015-04-24 17:31:52 -07001195 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001196}
1197
1198func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1199 // object files can't have any dynamic dependencies
1200 return nil
1201}
1202
Colin Cross0676e2d2015-04-24 17:39:18 -07001203func (*ccObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001204 // object files can't have any dynamic dependencies
1205 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001206}
1207
1208func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001209 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001210
Colin Cross97ba0732015-03-23 17:50:24 -07001211 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001212
1213 var outputFile string
1214 if len(objFiles) == 1 {
1215 outputFile = objFiles[0]
1216 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001217 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+objectExtension)
Colin Cross3f40fa42015-01-30 17:27:36 -08001218 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1219 }
1220
1221 c.out = outputFile
1222
1223 ctx.CheckbuildFile(outputFile)
1224}
1225
Colin Cross97ba0732015-03-23 17:50:24 -07001226func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001227 // Object files do not get installed.
1228}
1229
Colin Cross3f40fa42015-01-30 17:27:36 -08001230func (c *ccObject) outputFile() string {
1231 return c.out
1232}
1233
Dan Albertc3144b12015-04-28 18:17:56 -07001234var _ ccObjectProvider = (*ccObject)(nil)
1235
Colin Cross3f40fa42015-01-30 17:27:36 -08001236//
1237// Executables
1238//
1239
Colin Cross7d5136f2015-05-11 13:39:40 -07001240type CCBinaryProperties struct {
1241 // compile executable with -static
1242 Static_executable bool
1243
1244 // set the name of the output
1245 Stem string `android:"arch_variant"`
1246
1247 // append to the name of the output
1248 Suffix string `android:"arch_variant"`
1249
1250 // if set, add an extra objcopy --prefix-symbols= step
1251 Prefix_symbols string
1252}
1253
Colin Cross97ba0732015-03-23 17:50:24 -07001254type CCBinary struct {
Colin Crossfa138792015-04-24 17:31:52 -07001255 CCLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001256 out string
Colin Crossd350ecd2015-04-28 13:25:36 -07001257 installFile string
Colin Cross7d5136f2015-05-11 13:39:40 -07001258 BinaryProperties CCBinaryProperties
Colin Cross3f40fa42015-01-30 17:27:36 -08001259}
1260
Colin Crossed4cf0b2015-03-26 14:43:45 -07001261func (c *CCBinary) buildStatic() bool {
1262 return c.BinaryProperties.Static_executable
1263}
1264
1265func (c *CCBinary) buildShared() bool {
1266 return !c.BinaryProperties.Static_executable
1267}
1268
Colin Cross97ba0732015-03-23 17:50:24 -07001269func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001270 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001271 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001272 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001273 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001274
1275 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001276}
1277
Colin Cross0676e2d2015-04-24 17:39:18 -07001278func (c *CCBinary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001279 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001280 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001281 if c.Properties.Sdk_version == "" {
1282 if c.BinaryProperties.Static_executable {
1283 depNames.CrtBegin = "crtbegin_static"
1284 } else {
1285 depNames.CrtBegin = "crtbegin_dynamic"
1286 }
1287 depNames.CrtEnd = "crtend_android"
Colin Cross3f40fa42015-01-30 17:27:36 -08001288 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001289 if c.BinaryProperties.Static_executable {
1290 depNames.CrtBegin = "ndk_crtbegin_static." + c.Properties.Sdk_version
1291 } else {
1292 depNames.CrtBegin = "ndk_crtbegin_dynamic." + c.Properties.Sdk_version
1293 }
1294 depNames.CrtEnd = "ndk_crtend_android." + c.Properties.Sdk_version
Colin Cross3f40fa42015-01-30 17:27:36 -08001295 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001296
1297 if c.BinaryProperties.Static_executable {
Colin Cross74d1ec02015-04-28 13:30:13 -07001298 if c.stl(ctx) == "libc++_static" {
1299 depNames.StaticLibs = append(depNames.StaticLibs, "libm", "libc", "libdl")
1300 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001301 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1302 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1303 // move them to the beginning of deps.LateStaticLibs
1304 var groupLibs []string
1305 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1306 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1307 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1308 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001309 }
Colin Cross21b9a242015-03-24 14:15:58 -07001310 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001311}
1312
Colin Cross97ba0732015-03-23 17:50:24 -07001313func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001314 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001315
Colin Cross1f8f2342015-03-26 16:09:47 -07001316 props = append(props, &binary.BinaryProperties)
1317
Colin Crossfa138792015-04-24 17:31:52 -07001318 return newCCDynamic(&binary.CCLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001319}
1320
Colin Cross97ba0732015-03-23 17:50:24 -07001321func CCBinaryFactory() (blueprint.Module, []interface{}) {
1322 module := &CCBinary{}
1323
1324 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001325}
1326
Colin Cross18b6dc52015-04-28 13:20:37 -07001327func (c *CCBinary) ModifyProperties(ctx common.AndroidBaseContext) {
Colin Cross0af4b842015-04-30 16:36:18 -07001328 if ctx.Darwin() {
1329 c.BinaryProperties.Static_executable = false
1330 }
Colin Cross18b6dc52015-04-28 13:20:37 -07001331 if c.BinaryProperties.Static_executable {
1332 c.dynamicProperties.VariantIsStaticBinary = true
1333 }
1334}
1335
Colin Cross0676e2d2015-04-24 17:39:18 -07001336func (c *CCBinary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001337 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001338
Colin Cross97ba0732015-03-23 17:50:24 -07001339 flags.CFlags = append(flags.CFlags, "-fpie")
1340
Colin Crossf6566ed2015-03-24 11:13:38 -07001341 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001342 if c.BinaryProperties.Static_executable {
1343 // Clang driver needs -static to create static executable.
1344 // However, bionic/linker uses -shared to overwrite.
1345 // Linker for x86 targets does not allow coexistance of -static and -shared,
1346 // so we add -static only if -shared is not used.
1347 if !inList("-shared", flags.LdFlags) {
1348 flags.LdFlags = append(flags.LdFlags, "-static")
1349 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001350
Colin Crossed4cf0b2015-03-26 14:43:45 -07001351 flags.LdFlags = append(flags.LdFlags,
1352 "-nostdlib",
1353 "-Bstatic",
1354 "-Wl,--gc-sections",
1355 )
1356
1357 } else {
1358 linker := "/system/bin/linker"
1359 if flags.Toolchain.Is64Bit() {
1360 linker = "/system/bin/linker64"
1361 }
1362
1363 flags.LdFlags = append(flags.LdFlags,
1364 "-nostdlib",
1365 "-Bdynamic",
1366 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1367 "-Wl,--gc-sections",
1368 "-Wl,-z,nocopyreloc",
1369 )
1370 }
Colin Cross0af4b842015-04-30 16:36:18 -07001371 } else if ctx.Darwin() {
1372 flags.LdFlags = append(flags.LdFlags, "-Wl,-headerpad_max_install_names")
Colin Cross3f40fa42015-01-30 17:27:36 -08001373 }
1374
Colin Cross97ba0732015-03-23 17:50:24 -07001375 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001376}
1377
Colin Cross97ba0732015-03-23 17:50:24 -07001378func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1379 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001380
Colin Crossfa138792015-04-24 17:31:52 -07001381 if !c.BinaryProperties.Static_executable && inList("libc", c.Properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001382 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1383 "from static libs or set static_executable: true")
1384 }
1385
1386 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001387 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001388 if c.BinaryProperties.Prefix_symbols != "" {
1389 afterPrefixSymbols := outputFile
1390 outputFile = outputFile + ".intermediate"
1391 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1392 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1393 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001394
Colin Cross97ba0732015-03-23 17:50:24 -07001395 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001396 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001397 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001398}
Colin Cross3f40fa42015-01-30 17:27:36 -08001399
Colin Cross97ba0732015-03-23 17:50:24 -07001400func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossd350ecd2015-04-28 13:25:36 -07001401 c.installFile = ctx.InstallFile(filepath.Join("bin", c.Properties.Relative_install_path), c.out)
1402}
1403
1404func (c *CCBinary) HostToolPath() string {
1405 if c.HostOrDevice().Host() {
1406 return c.installFile
1407 }
1408 return ""
Dan Albertc403f7c2015-03-18 14:01:18 -07001409}
1410
Colin Cross7d5136f2015-05-11 13:39:40 -07001411type CCTestProperties struct {
1412 // Create a separate test for each source file. Useful when there is
1413 // global state that can not be torn down and reset between each test suite.
1414 Test_per_src bool
1415}
1416
Colin Cross9ffb4f52015-04-24 17:48:09 -07001417type CCTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001418 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001419
Colin Cross7d5136f2015-05-11 13:39:40 -07001420 TestProperties CCTestProperties
Dan Albertc403f7c2015-03-18 14:01:18 -07001421}
1422
Colin Cross9ffb4f52015-04-24 17:48:09 -07001423func (c *CCTest) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross0676e2d2015-04-24 17:39:18 -07001424 flags = c.CCBinary.flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001425
Colin Cross97ba0732015-03-23 17:50:24 -07001426 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001427 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001428 flags.CFlags = append(flags.CFlags, "-O0", "-g")
Colin Cross28344522015-04-22 13:07:53 -07001429 flags.LdFlags = append(flags.LdFlags, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001430 }
1431
1432 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross28344522015-04-22 13:07:53 -07001433 flags.CFlags = append(flags.CFlags,
1434 "-I"+filepath.Join(ctx.AConfig().SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001435
Colin Cross21b9a242015-03-24 14:15:58 -07001436 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001437}
1438
Colin Cross9ffb4f52015-04-24 17:48:09 -07001439func (c *CCTest) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001440 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
Colin Crossa8a93d32015-04-28 13:26:49 -07001441 depNames = c.CCBinary.depNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -07001442 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001443}
1444
Colin Cross9ffb4f52015-04-24 17:48:09 -07001445func (c *CCTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001446 if ctx.Device() {
Colin Crossa8a93d32015-04-28 13:26:49 -07001447 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001448 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001449 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001450 }
1451}
1452
Colin Cross9ffb4f52015-04-24 17:48:09 -07001453func (c *CCTest) testPerSrc() bool {
1454 return c.TestProperties.Test_per_src
Colin Cross6b290692015-03-19 14:05:33 -07001455}
1456
Colin Cross9ffb4f52015-04-24 17:48:09 -07001457func (c *CCTest) test() *CCTest {
1458 return c
1459}
1460
1461func NewCCTest(test *CCTest, module CCModuleType,
1462 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1463
1464 props = append(props, &test.TestProperties)
1465
1466 return NewCCBinary(&test.CCBinary, module, hod, props...)
1467}
1468
1469func CCTestFactory() (blueprint.Module, []interface{}) {
1470 module := &CCTest{}
1471
1472 return NewCCTest(module, module, common.HostAndDeviceSupported)
1473}
1474
1475type testPerSrc interface {
1476 test() *CCTest
1477 testPerSrc() bool
1478}
1479
1480var _ testPerSrc = (*CCTest)(nil)
1481
Colin Cross6b290692015-03-19 14:05:33 -07001482func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001483 if test, ok := mctx.Module().(testPerSrc); ok {
1484 if test.testPerSrc() {
1485 testNames := make([]string, len(test.test().Properties.Srcs))
1486 for i, src := range test.test().Properties.Srcs {
Colin Cross6b290692015-03-19 14:05:33 -07001487 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1488 }
1489 tests := mctx.CreateLocalVariations(testNames...)
Colin Cross9ffb4f52015-04-24 17:48:09 -07001490 for i, src := range test.test().Properties.Srcs {
1491 tests[i].(testPerSrc).test().Properties.Srcs = []string{src}
1492 tests[i].(testPerSrc).test().BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001493 }
1494 }
1495 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001496}
1497
Colin Cross2ba19d92015-05-07 15:44:20 -07001498type CCBenchmark struct {
1499 CCBinary
1500}
1501
1502func (c *CCBenchmark) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1503 depNames = c.CCBinary.depNames(ctx, depNames)
1504 depNames.StaticLibs = append(depNames.StaticLibs, "libbenchmark")
1505 return depNames
1506}
1507
1508func (c *CCBenchmark) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1509 if ctx.Device() {
1510 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
1511 } else {
1512 c.CCBinary.installModule(ctx, flags)
1513 }
1514}
1515
1516func NewCCBenchmark(test *CCBenchmark, module CCModuleType,
1517 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1518
1519 return NewCCBinary(&test.CCBinary, module, hod, props...)
1520}
1521
1522func CCBenchmarkFactory() (blueprint.Module, []interface{}) {
1523 module := &CCBenchmark{}
1524
1525 return NewCCBenchmark(module, module, common.HostAndDeviceSupported)
1526}
1527
Colin Cross3f40fa42015-01-30 17:27:36 -08001528//
1529// Static library
1530//
1531
Colin Cross97ba0732015-03-23 17:50:24 -07001532func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1533 module := &CCLibrary{}
1534 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001535
Colin Cross97ba0732015-03-23 17:50:24 -07001536 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001537}
1538
1539//
1540// Shared libraries
1541//
1542
Colin Cross97ba0732015-03-23 17:50:24 -07001543func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1544 module := &CCLibrary{}
1545 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001546
Colin Cross97ba0732015-03-23 17:50:24 -07001547 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001548}
1549
1550//
1551// Host static library
1552//
1553
Colin Cross97ba0732015-03-23 17:50:24 -07001554func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1555 module := &CCLibrary{}
1556 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001557
Colin Cross97ba0732015-03-23 17:50:24 -07001558 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001559}
1560
1561//
1562// Host Shared libraries
1563//
1564
Colin Cross97ba0732015-03-23 17:50:24 -07001565func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1566 module := &CCLibrary{}
1567 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001568
Colin Cross97ba0732015-03-23 17:50:24 -07001569 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001570}
1571
1572//
1573// Host Binaries
1574//
1575
Colin Cross97ba0732015-03-23 17:50:24 -07001576func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1577 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001578
Colin Cross97ba0732015-03-23 17:50:24 -07001579 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001580}
1581
1582//
Colin Cross1f8f2342015-03-26 16:09:47 -07001583// Host Tests
1584//
1585
1586func CCTestHostFactory() (blueprint.Module, []interface{}) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001587 module := &CCTest{}
Colin Cross1f8f2342015-03-26 16:09:47 -07001588 return NewCCBinary(&module.CCBinary, module, common.HostSupported,
Colin Cross9ffb4f52015-04-24 17:48:09 -07001589 &module.TestProperties)
Colin Cross1f8f2342015-03-26 16:09:47 -07001590}
1591
1592//
Colin Cross2ba19d92015-05-07 15:44:20 -07001593// Host Benchmarks
1594//
1595
1596func CCBenchmarkHostFactory() (blueprint.Module, []interface{}) {
1597 module := &CCBenchmark{}
1598 return NewCCBinary(&module.CCBinary, module, common.HostSupported)
1599}
1600
1601//
Colin Cross3f40fa42015-01-30 17:27:36 -08001602// Device libraries shipped with gcc
1603//
1604
1605type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001606 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001607}
1608
1609func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1610 // toolchain libraries can't have any dependencies
1611 return nil
1612}
1613
Colin Cross0676e2d2015-04-24 17:39:18 -07001614func (*toolchainLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001615 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001616 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001617}
1618
Colin Cross97ba0732015-03-23 17:50:24 -07001619func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001620 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001621
Colin Cross97ba0732015-03-23 17:50:24 -07001622 module.LibraryProperties.BuildStatic = true
1623
Colin Crossfa138792015-04-24 17:31:52 -07001624 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth,
Colin Cross21b9a242015-03-24 14:15:58 -07001625 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001626}
1627
1628func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001629 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001630
1631 libName := ctx.ModuleName() + staticLibraryExtension
1632 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1633
1634 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1635
1636 c.out = outputFile
1637
1638 ctx.CheckbuildFile(outputFile)
1639}
1640
Colin Cross97ba0732015-03-23 17:50:24 -07001641func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001642 // Toolchain libraries do not get installed.
1643}
1644
Dan Albertbe961682015-03-18 23:38:50 -07001645// NDK prebuilt libraries.
1646//
1647// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1648// either (with the exception of the shared STLs, which are installed to the app's directory rather
1649// than to the system image).
1650
1651func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1652 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
Colin Cross1332b002015-04-07 17:11:30 -07001653 ctx.AConfig().SrcDir(), version, toolchain.Name())
Dan Albertbe961682015-03-18 23:38:50 -07001654}
1655
Dan Albertc3144b12015-04-28 18:17:56 -07001656func ndkPrebuiltModuleToPath(ctx common.AndroidModuleContext, toolchain Toolchain,
1657 ext string, version string) string {
1658
1659 // NDK prebuilts are named like: ndk_NAME.EXT.SDK_VERSION.
1660 // We want to translate to just NAME.EXT
1661 name := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1662 dir := getNdkLibDir(ctx, toolchain, version)
1663 return filepath.Join(dir, name+ext)
1664}
1665
1666type ndkPrebuiltObject struct {
1667 ccObject
1668}
1669
1670func (*ndkPrebuiltObject) AndroidDynamicDependencies(
1671 ctx common.AndroidDynamicDependerModuleContext) []string {
1672
1673 // NDK objects can't have any dependencies
1674 return nil
1675}
1676
1677func (*ndkPrebuiltObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1678 // NDK objects can't have any dependencies
1679 return CCDeps{}
1680}
1681
1682func NdkPrebuiltObjectFactory() (blueprint.Module, []interface{}) {
1683 module := &ndkPrebuiltObject{}
1684 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
1685}
1686
1687func (c *ndkPrebuiltObject) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1688 deps CCDeps, objFiles []string) {
1689 // A null build step, but it sets up the output path.
1690 if !strings.HasPrefix(ctx.ModuleName(), "ndk_crt") {
1691 ctx.ModuleErrorf("NDK prebuilts must have an ndk_crt prefixed name")
1692 }
1693
1694 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, objectExtension, c.Properties.Sdk_version)
1695}
1696
1697func (c *ndkPrebuiltObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1698 // Objects do not get installed.
1699}
1700
1701var _ ccObjectProvider = (*ndkPrebuiltObject)(nil)
1702
Dan Albertbe961682015-03-18 23:38:50 -07001703type ndkPrebuiltLibrary struct {
1704 CCLibrary
1705}
1706
1707func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1708 ctx common.AndroidDynamicDependerModuleContext) []string {
1709
1710 // NDK libraries can't have any dependencies
1711 return nil
1712}
1713
Colin Cross0676e2d2015-04-24 17:39:18 -07001714func (*ndkPrebuiltLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Dan Albertbe961682015-03-18 23:38:50 -07001715 // NDK libraries can't have any dependencies
1716 return CCDeps{}
1717}
1718
1719func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1720 module := &ndkPrebuiltLibrary{}
1721 module.LibraryProperties.BuildShared = true
1722 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1723}
1724
1725func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1726 deps CCDeps, objFiles []string) {
1727 // A null build step, but it sets up the output path.
1728 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1729 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1730 }
1731
Colin Crossfa138792015-04-24 17:31:52 -07001732 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001733 c.exportFlags = []string{common.JoinWithPrefix(includeDirs, "-isystem ")}
Dan Albertbe961682015-03-18 23:38:50 -07001734
Dan Albertc3144b12015-04-28 18:17:56 -07001735 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, sharedLibraryExtension,
1736 c.Properties.Sdk_version)
Dan Albertbe961682015-03-18 23:38:50 -07001737}
1738
1739func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc3144b12015-04-28 18:17:56 -07001740 // NDK prebuilt libraries do not get installed.
Dan Albertbe961682015-03-18 23:38:50 -07001741}
1742
1743// The NDK STLs are slightly different from the prebuilt system libraries:
1744// * Are not specific to each platform version.
1745// * The libraries are not in a predictable location for each STL.
1746
1747type ndkPrebuiltStl struct {
1748 ndkPrebuiltLibrary
1749}
1750
1751type ndkPrebuiltStaticStl struct {
1752 ndkPrebuiltStl
1753}
1754
1755type ndkPrebuiltSharedStl struct {
1756 ndkPrebuiltStl
1757}
1758
1759func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1760 module := &ndkPrebuiltSharedStl{}
1761 module.LibraryProperties.BuildShared = true
1762 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1763}
1764
1765func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1766 module := &ndkPrebuiltStaticStl{}
1767 module.LibraryProperties.BuildStatic = true
1768 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1769}
1770
1771func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1772 gccVersion := toolchain.GccVersion()
1773 var libDir string
1774 switch stl {
1775 case "libstlport":
1776 libDir = "cxx-stl/stlport/libs"
1777 case "libc++":
1778 libDir = "cxx-stl/llvm-libc++/libs"
1779 case "libgnustl":
1780 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1781 }
1782
1783 if libDir != "" {
Colin Cross1332b002015-04-07 17:11:30 -07001784 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources"
Dan Albertbe961682015-03-18 23:38:50 -07001785 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1786 }
1787
1788 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1789 return ""
1790}
1791
1792func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1793 deps CCDeps, objFiles []string) {
1794 // A null build step, but it sets up the output path.
1795 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1796 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1797 }
1798
Colin Crossfa138792015-04-24 17:31:52 -07001799 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001800 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Dan Albertbe961682015-03-18 23:38:50 -07001801
1802 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1803 libExt := sharedLibraryExtension
1804 if c.LibraryProperties.BuildStatic {
1805 libExt = staticLibraryExtension
1806 }
1807
1808 stlName := strings.TrimSuffix(libName, "_shared")
1809 stlName = strings.TrimSuffix(stlName, "_static")
1810 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1811 c.out = libDir + "/" + libName + libExt
1812}
1813
Colin Cross3f40fa42015-01-30 17:27:36 -08001814func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001815 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001816 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001817 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001818 modules = mctx.CreateLocalVariations("static", "shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001819 modules[0].(ccLinkedInterface).setStatic(true)
1820 modules[1].(ccLinkedInterface).setStatic(false)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001821 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001822 modules = mctx.CreateLocalVariations("static")
Colin Cross18b6dc52015-04-28 13:20:37 -07001823 modules[0].(ccLinkedInterface).setStatic(true)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001824 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001825 modules = mctx.CreateLocalVariations("shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001826 modules[0].(ccLinkedInterface).setStatic(false)
Colin Cross3f40fa42015-01-30 17:27:36 -08001827 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001828 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001829 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001830
1831 if _, ok := c.(ccLibraryInterface); ok {
1832 reuseFrom := modules[0].(ccLibraryInterface)
1833 for _, m := range modules {
1834 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001835 }
1836 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001837 }
1838}
Colin Cross74d1ec02015-04-28 13:30:13 -07001839
1840// lastUniqueElements returns all unique elements of a slice, keeping the last copy of each
1841// modifies the slice contents in place, and returns a subslice of the original slice
1842func lastUniqueElements(list []string) []string {
1843 totalSkip := 0
1844 for i := len(list) - 1; i >= totalSkip; i-- {
1845 skip := 0
1846 for j := i - 1; j >= totalSkip; j-- {
1847 if list[i] == list[j] {
1848 skip++
1849 } else {
1850 list[j+skip] = list[j]
1851 }
1852 }
1853 totalSkip += skip
1854 }
1855 return list[totalSkip:]
1856}