blob: 2cc4ffb8a573ffd36faa13577da6d9d3db9ed06d [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.
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700157 Srcs []string `android:"arch_variant"`
158
159 // list of source files that should not be used to build the C/C++ module.
160 // This is most useful in the arch/multilib variants to remove non-common files
161 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700162
163 // list of module-specific flags that will be used for C and C++ compiles.
164 Cflags []string `android:"arch_variant"`
165
166 // list of module-specific flags that will be used for C++ compiles
167 Cppflags []string `android:"arch_variant"`
168
169 // list of module-specific flags that will be used for C compiles
170 Conlyflags []string `android:"arch_variant"`
171
172 // list of module-specific flags that will be used for .S compiles
173 Asflags []string `android:"arch_variant"`
174
175 // list of module-specific flags that will be used for .y and .yy compiles
176 Yaccflags []string
177
178 // list of module-specific flags that will be used for all link steps
179 Ldflags []string `android:"arch_variant"`
180
181 // the instruction set architecture to use to compile the C/C++
182 // module.
183 Instruction_set string `android:"arch_variant"`
184
185 // list of directories relative to the root of the source tree that will
186 // be added to the include path using -I.
187 // If possible, don't use this. If adding paths from the current directory use
188 // local_include_dirs, if adding paths from other modules use export_include_dirs in
189 // that module.
190 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
194 Local_include_dirs []string `android:"arch_variant"`
195
196 // list of directories relative to the Blueprints file that will
197 // be added to the include path using -I for any module that links against this module
198 Export_include_dirs []string `android:"arch_variant"`
199
200 // list of module-specific flags that will be used for C and C++ compiles when
201 // compiling with clang
202 Clang_cflags []string `android:"arch_variant"`
203
204 // list of module-specific flags that will be used for .S compiles when
205 // compiling with clang
206 Clang_asflags []string `android:"arch_variant"`
207
208 // list of system libraries that will be dynamically linked to
209 // shared library and executable modules. If unset, generally defaults to libc
210 // and libm. Set to [] to prevent linking against libc and libm.
211 System_shared_libs []string
212
213 // list of modules whose object files should be linked into this module
214 // in their entirety. For static library modules, all of the .o files from the intermediate
215 // directory of the dependency will be linked into this modules .a file. For a shared library,
216 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
217 Whole_static_libs []string `android:"arch_variant"`
218
219 // list of modules that should be statically linked into this module.
220 Static_libs []string `android:"arch_variant"`
221
222 // list of modules that should be dynamically linked into this module.
223 Shared_libs []string `android:"arch_variant"`
224
225 // allow the module to contain undefined symbols. By default,
226 // modules cannot contain undefined symbols that are not satisified by their immediate
227 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
228 // This flag should only be necessary for compiling low-level libraries like libc.
229 Allow_undefined_symbols bool
230
231 // don't link in crt_begin and crt_end. This flag should only be necessary for
232 // compiling crt or libc.
233 Nocrt bool `android:"arch_variant"`
234
235 // don't insert default compiler flags into asflags, cflags,
236 // cppflags, conlyflags, ldflags, or include_dirs
237 No_default_compiler_flags bool
238
239 // compile module with clang instead of gcc
240 Clang bool `android:"arch_variant"`
241
242 // pass -frtti instead of -fno-rtti
243 Rtti bool
244
245 // -l arguments to pass to linker for host-provided shared libraries
246 Host_ldlibs []string `android:"arch_variant"`
247
248 // select the STL library to use. Possible values are "libc++", "libc++_static",
249 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
250 // default
251 Stl string
252
253 // Set for combined shared/static libraries to prevent compiling object files a second time
254 SkipCompileObjs bool `blueprint:"mutated"`
255
256 Debug, Release struct {
257 // list of module-specific flags that will be used for C and C++ compiles in debug or
258 // release builds
259 Cflags []string `android:"arch_variant"`
260 } `android:"arch_variant"`
261
262 // Minimum sdk version supported when compiling against the ndk
263 Sdk_version string
264
265 // install to a subdirectory of the default install path for the module
266 Relative_install_path string
267}
268
Colin Crossfa138792015-04-24 17:31:52 -0700269// CCBase contains the properties and members used by all C/C++ module types, and implements
Colin Crossc472d572015-03-17 15:06:21 -0700270// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
271// and uses a ccModuleType interface to that struct to create the build steps.
Colin Crossfa138792015-04-24 17:31:52 -0700272type CCBase struct {
Colin Crossc472d572015-03-17 15:06:21 -0700273 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700274 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700275
Colin Cross7d5136f2015-05-11 13:39:40 -0700276 Properties CCBaseProperties
Colin Crossfa138792015-04-24 17:31:52 -0700277
278 unused struct {
279 Asan bool
280 Native_coverage bool
281 Strip string
282 Tags []string
283 Required []string
284 }
Colin Crossc472d572015-03-17 15:06:21 -0700285
286 installPath string
Colin Cross74d1ec02015-04-28 13:30:13 -0700287
288 savedDepNames CCDeps
Colin Crossc472d572015-03-17 15:06:21 -0700289}
290
Colin Crossfa138792015-04-24 17:31:52 -0700291func newCCBase(base *CCBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700292 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
293
294 base.module = module
295
Colin Crossfa138792015-04-24 17:31:52 -0700296 props = append(props, &base.Properties, &base.unused)
Colin Crossc472d572015-03-17 15:06:21 -0700297
Colin Cross5049f022015-03-18 13:28:46 -0700298 return common.InitAndroidArchModule(module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700299}
300
Colin Crossfa138792015-04-24 17:31:52 -0700301func (c *CCBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800302 toolchain := c.findToolchain(ctx)
303 if ctx.Failed() {
304 return
305 }
306
Colin Cross21b9a242015-03-24 14:15:58 -0700307 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800308 if ctx.Failed() {
309 return
310 }
311
Colin Cross74d1ec02015-04-28 13:30:13 -0700312 deps := c.depsToPaths(ctx, c.savedDepNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800313 if ctx.Failed() {
314 return
315 }
316
Colin Cross28344522015-04-22 13:07:53 -0700317 flags.CFlags = append(flags.CFlags, deps.Cflags...)
Colin Crossed9f8682015-03-18 17:17:35 -0700318
Colin Cross581c1892015-04-07 16:50:10 -0700319 objFiles := c.compileObjs(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800320 if ctx.Failed() {
321 return
322 }
323
Colin Cross581c1892015-04-07 16:50:10 -0700324 generatedObjFiles := c.compileGeneratedObjs(ctx, flags)
Colin Cross5049f022015-03-18 13:28:46 -0700325 if ctx.Failed() {
326 return
327 }
328
329 objFiles = append(objFiles, generatedObjFiles...)
330
Colin Cross3f40fa42015-01-30 17:27:36 -0800331 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
332 if ctx.Failed() {
333 return
334 }
Dan Albertc403f7c2015-03-18 14:01:18 -0700335
336 c.ccModuleType().installModule(ctx, flags)
337 if ctx.Failed() {
338 return
339 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800340}
341
Colin Crossfa138792015-04-24 17:31:52 -0700342func (c *CCBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800343 return c.module
344}
345
Colin Crossfa138792015-04-24 17:31:52 -0700346var _ common.AndroidDynamicDepender = (*CCBase)(nil)
Colin Cross3f40fa42015-01-30 17:27:36 -0800347
Colin Crossfa138792015-04-24 17:31:52 -0700348func (c *CCBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800349 arch := ctx.Arch()
Colin Crossd3ba0392015-05-07 14:11:29 -0700350 hod := ctx.HostOrDevice()
351 factory := toolchainFactories[hod][arch.ArchType]
Colin Cross3f40fa42015-01-30 17:27:36 -0800352 if factory == nil {
353 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
Colin Crossd3ba0392015-05-07 14:11:29 -0700354 hod.String(), arch.String()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800355 }
356 return factory(arch.ArchVariant, arch.CpuVariant)
357}
358
Colin Crossfa138792015-04-24 17:31:52 -0700359func (c *CCBase) ModifyProperties(ctx common.AndroidBaseContext) {
360}
361
Colin Crosse11befc2015-04-27 17:49:17 -0700362func (c *CCBase) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossfa138792015-04-24 17:31:52 -0700363 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.Properties.Whole_static_libs...)
364 depNames.StaticLibs = append(depNames.StaticLibs, c.Properties.Static_libs...)
365 depNames.SharedLibs = append(depNames.SharedLibs, c.Properties.Shared_libs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700366
Colin Cross21b9a242015-03-24 14:15:58 -0700367 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800368}
369
Colin Crossfa138792015-04-24 17:31:52 -0700370func (c *CCBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
371 c.module.ModifyProperties(ctx)
372
Colin Cross74d1ec02015-04-28 13:30:13 -0700373 c.savedDepNames = c.module.depNames(ctx, CCDeps{})
374 c.savedDepNames.WholeStaticLibs = lastUniqueElements(c.savedDepNames.WholeStaticLibs)
375 c.savedDepNames.StaticLibs = lastUniqueElements(c.savedDepNames.StaticLibs)
376 c.savedDepNames.SharedLibs = lastUniqueElements(c.savedDepNames.SharedLibs)
377
378 staticLibs := c.savedDepNames.WholeStaticLibs
379 staticLibs = append(staticLibs, c.savedDepNames.StaticLibs...)
380 staticLibs = append(staticLibs, c.savedDepNames.LateStaticLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700381 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800382
Colin Cross74d1ec02015-04-28 13:30:13 -0700383 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, c.savedDepNames.SharedLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700384
Colin Cross74d1ec02015-04-28 13:30:13 -0700385 ret := append([]string(nil), c.savedDepNames.ObjFiles...)
386 if c.savedDepNames.CrtBegin != "" {
387 ret = append(ret, c.savedDepNames.CrtBegin)
Colin Cross21b9a242015-03-24 14:15:58 -0700388 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700389 if c.savedDepNames.CrtEnd != "" {
390 ret = append(ret, c.savedDepNames.CrtEnd)
Colin Cross21b9a242015-03-24 14:15:58 -0700391 }
392
393 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800394}
395
396// Create a ccFlags struct that collects the compile flags from global values,
397// per-target values, module type values, and per-module Blueprints properties
Colin Crossfa138792015-04-24 17:31:52 -0700398func (c *CCBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700399 flags := CCFlags{
Colin Crossfa138792015-04-24 17:31:52 -0700400 CFlags: c.Properties.Cflags,
401 CppFlags: c.Properties.Cppflags,
402 ConlyFlags: c.Properties.Conlyflags,
403 LdFlags: c.Properties.Ldflags,
404 AsFlags: c.Properties.Asflags,
405 YaccFlags: c.Properties.Yaccflags,
406 Nocrt: c.Properties.Nocrt,
Colin Cross97ba0732015-03-23 17:50:24 -0700407 Toolchain: toolchain,
Colin Crossfa138792015-04-24 17:31:52 -0700408 Clang: c.Properties.Clang,
Colin Cross3f40fa42015-01-30 17:27:36 -0800409 }
Colin Cross28344522015-04-22 13:07:53 -0700410
411 // Include dir cflags
Colin Crossf2298272015-05-12 11:36:53 -0700412 common.CheckSrcDirsExist(ctx, c.Properties.Include_dirs, "include_dirs")
413 common.CheckModuleSrcDirsExist(ctx, c.Properties.Local_include_dirs, "local_include_dirs")
414
Colin Crossfa138792015-04-24 17:31:52 -0700415 rootIncludeDirs := pathtools.PrefixPaths(c.Properties.Include_dirs, ctx.AConfig().SrcDir())
416 localIncludeDirs := pathtools.PrefixPaths(c.Properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -0700417 flags.GlobalFlags = append(flags.GlobalFlags,
418 includeDirsToFlags(rootIncludeDirs),
419 includeDirsToFlags(localIncludeDirs))
420
Colin Crossfa138792015-04-24 17:31:52 -0700421 if !c.Properties.No_default_compiler_flags {
422 if c.Properties.Sdk_version == "" || ctx.Host() {
Colin Cross28344522015-04-22 13:07:53 -0700423 flags.GlobalFlags = append(flags.GlobalFlags,
424 "${commonGlobalIncludes}",
425 toolchain.IncludeFlags(),
426 "-I${SrcDir}/libnativehelper/include/nativehelper")
427 }
428
429 flags.GlobalFlags = append(flags.GlobalFlags, []string{
430 "-I" + common.ModuleSrcDir(ctx),
431 "-I" + common.ModuleOutDir(ctx),
432 "-I" + common.ModuleGenDir(ctx),
433 }...)
434 }
435
Colin Crossfa138792015-04-24 17:31:52 -0700436 instructionSet := c.Properties.Instruction_set
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700437 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
438 if err != nil {
439 ctx.ModuleErrorf("%s", err)
440 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800441
Colin Crossaf19a292015-03-18 12:07:10 -0700442 // TODO: debug
Colin Crossfa138792015-04-24 17:31:52 -0700443 flags.CFlags = append(flags.CFlags, c.Properties.Release.Cflags...)
Colin Crossaf19a292015-03-18 12:07:10 -0700444
Colin Cross28d76592015-03-26 16:14:04 -0700445 if ctx.Host() && !ctx.ContainsProperty("clang") {
Colin Cross97ba0732015-03-23 17:50:24 -0700446 flags.Clang = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800447 }
448
Colin Cross97ba0732015-03-23 17:50:24 -0700449 if flags.Clang {
450 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
Colin Crossfa138792015-04-24 17:31:52 -0700451 flags.CFlags = append(flags.CFlags, c.Properties.Clang_cflags...)
452 flags.AsFlags = append(flags.AsFlags, c.Properties.Clang_asflags...)
Colin Cross97ba0732015-03-23 17:50:24 -0700453 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
454 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
455 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800456
Colin Cross97ba0732015-03-23 17:50:24 -0700457 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
458 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700459 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700460 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700461 }
462
Colin Cross3f40fa42015-01-30 17:27:36 -0800463 target := "-target " + toolchain.ClangTriple()
464 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
465
Colin Cross97ba0732015-03-23 17:50:24 -0700466 flags.CFlags = append(flags.CFlags, target, gccPrefix)
467 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
468 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800469 }
470
Colin Crossfa138792015-04-24 17:31:52 -0700471 if !c.Properties.No_default_compiler_flags {
472 if ctx.Device() && !c.Properties.Allow_undefined_symbols {
Colin Cross97ba0732015-03-23 17:50:24 -0700473 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800474 }
475
Colin Cross56b4d452015-04-21 17:38:44 -0700476 flags.GlobalFlags = append(flags.GlobalFlags, instructionSetFlags)
477
Colin Cross97ba0732015-03-23 17:50:24 -0700478 if flags.Clang {
479 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700480 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800481 toolchain.ClangCflags(),
482 "${commonClangGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700483 fmt.Sprintf("${%sClangGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800484 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700485 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700486 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800487 toolchain.Cflags(),
488 "${commonGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700489 fmt.Sprintf("${%sGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800490 }
491
Colin Crossf6566ed2015-03-24 11:13:38 -0700492 if ctx.Device() {
Colin Crossfa138792015-04-24 17:31:52 -0700493 if c.Properties.Rtti {
Colin Cross97ba0732015-03-23 17:50:24 -0700494 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800495 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700496 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800497 }
498 }
499
Colin Cross97ba0732015-03-23 17:50:24 -0700500 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800501
Colin Cross97ba0732015-03-23 17:50:24 -0700502 if flags.Clang {
503 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
504 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800505 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700506 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
507 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800508 }
Colin Cross28344522015-04-22 13:07:53 -0700509
510 if ctx.Host() {
Colin Crossfa138792015-04-24 17:31:52 -0700511 flags.LdFlags = append(flags.LdFlags, c.Properties.Host_ldlibs...)
Colin Cross28344522015-04-22 13:07:53 -0700512 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800513 }
514
Colin Cross0676e2d2015-04-24 17:39:18 -0700515 flags = c.ccModuleType().flags(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800516
517 // Optimization to reduce size of build.ninja
518 // Replace the long list of flags for each file with a module-local variable
Colin Cross97ba0732015-03-23 17:50:24 -0700519 ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " "))
520 ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " "))
521 ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " "))
522 flags.CFlags = []string{"$cflags"}
523 flags.CppFlags = []string{"$cppflags"}
524 flags.AsFlags = []string{"$asflags"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800525
526 return flags
527}
528
Colin Cross0676e2d2015-04-24 17:39:18 -0700529func (c *CCBase) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross3f40fa42015-01-30 17:27:36 -0800530 return flags
531}
532
533// Compile a list of source files into objects a specified subdirectory
Colin Crossfa138792015-04-24 17:31:52 -0700534func (c *CCBase) customCompileObjs(ctx common.AndroidModuleContext, flags CCFlags,
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700535 subdir string, srcFiles, excludes []string) []string {
Colin Cross581c1892015-04-07 16:50:10 -0700536
537 buildFlags := ccFlagsToBuilderFlags(flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800538
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700539 srcFiles = ctx.ExpandSources(srcFiles, excludes)
Colin Cross581c1892015-04-07 16:50:10 -0700540 srcFiles, deps := genSources(ctx, srcFiles, buildFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800541
Colin Cross581c1892015-04-07 16:50:10 -0700542 return TransformSourceToObj(ctx, subdir, srcFiles, buildFlags, deps)
Colin Cross3f40fa42015-01-30 17:27:36 -0800543}
544
Colin Crossfa138792015-04-24 17:31:52 -0700545// Compile files listed in c.Properties.Srcs into objects
546func (c *CCBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800547
Colin Crossfa138792015-04-24 17:31:52 -0700548 if c.Properties.SkipCompileObjs {
Colin Cross3f40fa42015-01-30 17:27:36 -0800549 return nil
550 }
551
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700552 return c.customCompileObjs(ctx, flags, "", c.Properties.Srcs, c.Properties.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800553}
554
Colin Cross5049f022015-03-18 13:28:46 -0700555// Compile generated source files from dependencies
Colin Crossfa138792015-04-24 17:31:52 -0700556func (c *CCBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700557 var srcs []string
558
Colin Crossfa138792015-04-24 17:31:52 -0700559 if c.Properties.SkipCompileObjs {
Colin Cross5049f022015-03-18 13:28:46 -0700560 return nil
561 }
562
563 ctx.VisitDirectDeps(func(module blueprint.Module) {
564 if gen, ok := module.(genrule.SourceFileGenerator); ok {
565 srcs = append(srcs, gen.GeneratedSourceFiles()...)
566 }
567 })
568
569 if len(srcs) == 0 {
570 return nil
571 }
572
Colin Cross581c1892015-04-07 16:50:10 -0700573 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags), nil)
Colin Cross5049f022015-03-18 13:28:46 -0700574}
575
Colin Crossfa138792015-04-24 17:31:52 -0700576func (c *CCBase) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800577 return ""
578}
579
Colin Crossfa138792015-04-24 17:31:52 -0700580func (c *CCBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800581 names []string) (modules []common.AndroidModule,
Colin Cross28344522015-04-22 13:07:53 -0700582 outputFiles []string, exportedFlags []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800583
584 for _, n := range names {
585 found := false
586 ctx.VisitDirectDeps(func(m blueprint.Module) {
587 otherName := ctx.OtherModuleName(m)
588 if otherName != n {
589 return
590 }
591
Colin Cross97ba0732015-03-23 17:50:24 -0700592 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800593 if a.Disabled() {
594 // If a cc_library host+device module depends on a library that exists as both
595 // cc_library_shared and cc_library_host_shared, it will end up with two
596 // dependencies with the same name, one of which is marked disabled for each
597 // of host and device. Ignore the disabled one.
598 return
599 }
Colin Crossd3ba0392015-05-07 14:11:29 -0700600 if a.HostOrDevice() != ctx.HostOrDevice() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800601 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
602 otherName)
603 return
604 }
605
606 if outputFile := a.outputFile(); outputFile != "" {
607 if found {
608 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
609 return
610 }
611 outputFiles = append(outputFiles, outputFile)
612 modules = append(modules, a)
Colin Cross28344522015-04-22 13:07:53 -0700613 if i, ok := a.(ccExportedFlagsProducer); ok {
614 exportedFlags = append(exportedFlags, i.exportedFlags()...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800615 }
616 found = true
617 } else {
618 ctx.ModuleErrorf("module %q missing output file", otherName)
619 return
620 }
621 } else {
622 ctx.ModuleErrorf("module %q not an android module", otherName)
623 return
624 }
625 })
626 if !found {
627 ctx.ModuleErrorf("unsatisified dependency on %q", n)
628 }
629 }
630
Colin Cross28344522015-04-22 13:07:53 -0700631 return modules, outputFiles, exportedFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800632}
633
Colin Cross21b9a242015-03-24 14:15:58 -0700634// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
635// containing paths
Colin Crossfa138792015-04-24 17:31:52 -0700636func (c *CCBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700637 var depPaths CCDeps
Colin Cross28344522015-04-22 13:07:53 -0700638 var newCflags []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800639
Colin Cross21b9a242015-03-24 14:15:58 -0700640 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800641
Colin Cross28344522015-04-22 13:07:53 -0700642 wholeStaticLibModules, depPaths.WholeStaticLibs, newCflags =
Colin Cross21b9a242015-03-24 14:15:58 -0700643 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
Colin Cross28344522015-04-22 13:07:53 -0700644 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800645
Colin Cross21b9a242015-03-24 14:15:58 -0700646 for _, m := range wholeStaticLibModules {
647 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
648 depPaths.WholeStaticLibObjFiles =
649 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
650 } else {
651 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
652 }
653 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800654
Colin Cross28344522015-04-22 13:07:53 -0700655 _, depPaths.StaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.StaticLibs)
656 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700657
Colin Cross28344522015-04-22 13:07:53 -0700658 _, depPaths.LateStaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
659 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700660
Colin Cross28344522015-04-22 13:07:53 -0700661 _, depPaths.SharedLibs, newCflags = c.depsToPathsFromList(ctx, depNames.SharedLibs)
662 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700663
664 ctx.VisitDirectDeps(func(m blueprint.Module) {
Dan Albertc3144b12015-04-28 18:17:56 -0700665 if obj, ok := m.(ccObjectProvider); ok {
Colin Cross21b9a242015-03-24 14:15:58 -0700666 otherName := ctx.OtherModuleName(m)
667 if otherName == depNames.CrtBegin {
Colin Crossfa138792015-04-24 17:31:52 -0700668 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700669 depPaths.CrtBegin = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700670 }
671 } else if otherName == depNames.CrtEnd {
Colin Crossfa138792015-04-24 17:31:52 -0700672 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700673 depPaths.CrtEnd = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700674 }
675 } else {
Dan Albertc3144b12015-04-28 18:17:56 -0700676 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.object().outputFile())
Colin Cross21b9a242015-03-24 14:15:58 -0700677 }
678 }
679 })
680
681 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800682}
683
Colin Cross7d5136f2015-05-11 13:39:40 -0700684type ccLinkedProperties struct {
685 VariantIsShared bool `blueprint:"mutated"`
686 VariantIsStatic bool `blueprint:"mutated"`
687 VariantIsStaticBinary bool `blueprint:"mutated"`
688}
689
Colin Crossfa138792015-04-24 17:31:52 -0700690// CCLinked contains the properties and members used by libraries and executables
691type CCLinked struct {
692 CCBase
Colin Cross7d5136f2015-05-11 13:39:40 -0700693 dynamicProperties ccLinkedProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800694}
695
Colin Crossfa138792015-04-24 17:31:52 -0700696func newCCDynamic(dynamic *CCLinked, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700697 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
698
Colin Crossed4cf0b2015-03-26 14:43:45 -0700699 props = append(props, &dynamic.dynamicProperties)
700
Colin Crossfa138792015-04-24 17:31:52 -0700701 return newCCBase(&dynamic.CCBase, module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700702}
703
Colin Crossfa138792015-04-24 17:31:52 -0700704func (c *CCLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross28d76592015-03-26 16:14:04 -0700705 if ctx.ContainsProperty("system_shared_libs") {
Colin Crossfa138792015-04-24 17:31:52 -0700706 return c.Properties.System_shared_libs
707 } else if ctx.Device() && c.Properties.Sdk_version == "" {
Colin Cross577f6e42015-03-27 18:23:34 -0700708 return []string{"libc", "libm"}
Colin Cross28d76592015-03-26 16:14:04 -0700709 } else {
Colin Cross577f6e42015-03-27 18:23:34 -0700710 return nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800711 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800712}
713
Colin Crossfa138792015-04-24 17:31:52 -0700714func (c *CCLinked) stl(ctx common.AndroidBaseContext) string {
715 if c.Properties.Sdk_version != "" && ctx.Device() {
716 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700717 case "":
718 return "ndk_system"
719 case "c++_shared", "c++_static",
720 "stlport_shared", "stlport_static",
721 "gnustl_static":
Colin Crossfa138792015-04-24 17:31:52 -0700722 return "ndk_lib" + c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700723 default:
Colin Crossfa138792015-04-24 17:31:52 -0700724 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700725 return ""
726 }
727 }
728
Colin Crossfa138792015-04-24 17:31:52 -0700729 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700730 case "libc++", "libc++_static",
731 "stlport", "stlport_static",
732 "libstdc++":
Colin Crossfa138792015-04-24 17:31:52 -0700733 return c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700734 case "none":
735 return ""
736 case "":
Colin Cross18b6dc52015-04-28 13:20:37 -0700737 if c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700738 return "libc++_static"
Colin Cross18b6dc52015-04-28 13:20:37 -0700739 } else {
740 return "libc++" // TODO: mingw needs libstdc++
Colin Crossed4cf0b2015-03-26 14:43:45 -0700741 }
742 default:
Colin Crossfa138792015-04-24 17:31:52 -0700743 ctx.ModuleErrorf("stl: %q is not a supported STL", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700744 return ""
745 }
746}
747
Colin Cross0af4b842015-04-30 16:36:18 -0700748var hostDynamicGccLibs, hostStaticGccLibs []string
749
750func init() {
751 if runtime.GOOS == "darwin" {
752 hostDynamicGccLibs = []string{"-lc", "-lSystem"}
753 hostStaticGccLibs = []string{"NO_STATIC_HOST_BINARIES_ON_DARWIN"}
754 } else {
755 hostDynamicGccLibs = []string{"-lgcc_s", "-lgcc", "-lc", "-lgcc_s", "-lgcc"}
756 hostStaticGccLibs = []string{"-Wl,--start-group", "-lgcc", "-lgcc_eh", "-lc", "-Wl,--end-group"}
757 }
758}
Colin Cross712fc022015-04-27 11:13:34 -0700759
Colin Crosse11befc2015-04-27 17:49:17 -0700760func (c *CCLinked) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700761 stl := c.stl(ctx)
762 if ctx.Failed() {
763 return flags
764 }
765
766 switch stl {
767 case "libc++", "libc++_static":
768 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
Colin Cross28344522015-04-22 13:07:53 -0700769 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/external/libcxx/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700770 if ctx.Host() {
771 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
772 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross712fc022015-04-27 11:13:34 -0700773 flags.LdFlags = append(flags.LdFlags, "-lm", "-lpthread")
Colin Cross18b6dc52015-04-28 13:20:37 -0700774 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700775 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700776 } else {
777 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700778 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700779 }
780 case "stlport", "stlport_static":
781 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700782 flags.CFlags = append(flags.CFlags,
783 "-I${SrcDir}/external/stlport/stlport",
784 "-I${SrcDir}/bionic/libstdc++/include",
785 "-I${SrcDir}/bionic")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700786 }
787 case "libstdc++":
788 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
789 // tree is in good enough shape to not need it.
790 // Host builds will use GNU libstdc++.
791 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700792 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/bionic/libstdc++/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700793 }
794 case "ndk_system":
Colin Cross1332b002015-04-07 17:11:30 -0700795 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources/"
Colin Cross28344522015-04-22 13:07:53 -0700796 flags.CFlags = append(flags.CFlags, "-isystem "+ndkSrcRoot+"cxx-stl/system/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700797 case "ndk_libc++_shared", "ndk_libc++_static":
798 // TODO(danalbert): This really shouldn't be here...
799 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
800 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
801 // Nothing
802 case "":
803 // None or error.
804 if ctx.Host() {
805 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
806 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross18b6dc52015-04-28 13:20:37 -0700807 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700808 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700809 } else {
810 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700811 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700812 }
813 default:
Colin Crossfa138792015-04-24 17:31:52 -0700814 panic(fmt.Errorf("Unknown stl in CCLinked.Flags: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700815 }
816
817 return flags
818}
819
Colin Crosse11befc2015-04-27 17:49:17 -0700820func (c *CCLinked) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
821 depNames = c.CCBase.depNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800822
Colin Crossed4cf0b2015-03-26 14:43:45 -0700823 stl := c.stl(ctx)
824 if ctx.Failed() {
825 return depNames
826 }
827
828 switch stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700829 case "libstdc++":
830 if ctx.Device() {
831 depNames.SharedLibs = append(depNames.SharedLibs, stl)
832 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700833 case "libc++", "libc++_static":
834 if stl == "libc++" {
835 depNames.SharedLibs = append(depNames.SharedLibs, stl)
836 } else {
837 depNames.StaticLibs = append(depNames.StaticLibs, stl)
838 }
839 if ctx.Device() {
840 if ctx.Arch().ArchType == common.Arm {
841 depNames.StaticLibs = append(depNames.StaticLibs, "libunwind_llvm")
842 }
843 if c.staticBinary() {
844 depNames.StaticLibs = append(depNames.StaticLibs, "libdl")
845 } else {
846 depNames.SharedLibs = append(depNames.SharedLibs, "libdl")
847 }
848 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700849 case "stlport":
850 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
851 case "stlport_static":
852 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
853 case "":
854 // None or error.
855 case "ndk_system":
856 // TODO: Make a system STL prebuilt for the NDK.
857 // 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 -0700858 // its own includes. The includes are handled in CCBase.Flags().
Colin Cross577f6e42015-03-27 18:23:34 -0700859 depNames.SharedLibs = append([]string{"libstdc++"}, depNames.SharedLibs...)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700860 case "ndk_libc++_shared", "ndk_libstlport_shared":
861 depNames.SharedLibs = append(depNames.SharedLibs, stl)
862 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
863 depNames.StaticLibs = append(depNames.StaticLibs, stl)
864 default:
Colin Crosse11befc2015-04-27 17:49:17 -0700865 panic(fmt.Errorf("Unknown stl in CCLinked.depNames: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700866 }
867
Colin Cross74d1ec02015-04-28 13:30:13 -0700868 if ctx.ModuleName() != "libcompiler_rt-extras" {
869 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
870 }
871
Colin Crossf6566ed2015-03-24 11:13:38 -0700872 if ctx.Device() {
Colin Cross77b00fa2015-03-16 16:15:49 -0700873 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700874 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700875
Colin Cross18b6dc52015-04-28 13:20:37 -0700876 if !c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700877 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
878 }
Colin Cross577f6e42015-03-27 18:23:34 -0700879
Colin Crossfa138792015-04-24 17:31:52 -0700880 if c.Properties.Sdk_version != "" {
881 version := c.Properties.Sdk_version
Colin Cross577f6e42015-03-27 18:23:34 -0700882 depNames.SharedLibs = append(depNames.SharedLibs,
883 "ndk_libc."+version,
884 "ndk_libm."+version,
885 )
886 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800887 }
888
Colin Cross21b9a242015-03-24 14:15:58 -0700889 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800890}
891
Colin Crossed4cf0b2015-03-26 14:43:45 -0700892// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
893type ccLinkedInterface interface {
894 // Returns true if the build options for the module have selected a static or shared build
895 buildStatic() bool
896 buildShared() bool
897
898 // Sets whether a specific variant is static or shared
Colin Cross18b6dc52015-04-28 13:20:37 -0700899 setStatic(bool)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700900
Colin Cross18b6dc52015-04-28 13:20:37 -0700901 // Returns whether a specific variant is a static library or binary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700902 static() bool
Colin Cross18b6dc52015-04-28 13:20:37 -0700903
904 // Returns whether a module is a static binary
905 staticBinary() bool
Colin Crossed4cf0b2015-03-26 14:43:45 -0700906}
907
908var _ ccLinkedInterface = (*CCLibrary)(nil)
909var _ ccLinkedInterface = (*CCBinary)(nil)
910
Colin Crossfa138792015-04-24 17:31:52 -0700911func (c *CCLinked) static() bool {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700912 return c.dynamicProperties.VariantIsStatic
913}
914
Colin Cross18b6dc52015-04-28 13:20:37 -0700915func (c *CCLinked) staticBinary() bool {
916 return c.dynamicProperties.VariantIsStaticBinary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700917}
918
Colin Cross18b6dc52015-04-28 13:20:37 -0700919func (c *CCLinked) setStatic(static bool) {
920 c.dynamicProperties.VariantIsStatic = static
Colin Crossed4cf0b2015-03-26 14:43:45 -0700921}
922
Colin Cross28344522015-04-22 13:07:53 -0700923type ccExportedFlagsProducer interface {
924 exportedFlags() []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800925}
926
927//
928// Combined static+shared libraries
929//
930
Colin Cross7d5136f2015-05-11 13:39:40 -0700931type CCLibraryProperties struct {
932 BuildStatic bool `blueprint:"mutated"`
933 BuildShared bool `blueprint:"mutated"`
934 Static struct {
935 Srcs []string `android:"arch_variant"`
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700936 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700937 Cflags []string `android:"arch_variant"`
938 Whole_static_libs []string `android:"arch_variant"`
939 Static_libs []string `android:"arch_variant"`
940 Shared_libs []string `android:"arch_variant"`
941 } `android:"arch_variant"`
942 Shared struct {
943 Srcs []string `android:"arch_variant"`
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700944 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700945 Cflags []string `android:"arch_variant"`
946 Whole_static_libs []string `android:"arch_variant"`
947 Static_libs []string `android:"arch_variant"`
948 Shared_libs []string `android:"arch_variant"`
949 } `android:"arch_variant"`
950}
951
Colin Cross97ba0732015-03-23 17:50:24 -0700952type CCLibrary struct {
Colin Crossfa138792015-04-24 17:31:52 -0700953 CCLinked
Colin Cross3f40fa42015-01-30 17:27:36 -0800954
Colin Cross28344522015-04-22 13:07:53 -0700955 reuseFrom ccLibraryInterface
956 reuseObjFiles []string
957 objFiles []string
958 exportFlags []string
959 out string
Colin Cross3f40fa42015-01-30 17:27:36 -0800960
Colin Cross7d5136f2015-05-11 13:39:40 -0700961 LibraryProperties CCLibraryProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800962}
963
Colin Crossed4cf0b2015-03-26 14:43:45 -0700964func (c *CCLibrary) buildStatic() bool {
965 return c.LibraryProperties.BuildStatic
966}
967
968func (c *CCLibrary) buildShared() bool {
969 return c.LibraryProperties.BuildShared
970}
971
Colin Cross97ba0732015-03-23 17:50:24 -0700972type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700973 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -0700974 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700975 setReuseFrom(ccLibraryInterface)
976 getReuseFrom() ccLibraryInterface
977 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -0700978 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -0700979}
980
Colin Crossed4cf0b2015-03-26 14:43:45 -0700981var _ ccLibraryInterface = (*CCLibrary)(nil)
982
Colin Cross97ba0732015-03-23 17:50:24 -0700983func (c *CCLibrary) ccLibrary() *CCLibrary {
984 return c
Colin Cross3f40fa42015-01-30 17:27:36 -0800985}
986
Colin Cross97ba0732015-03-23 17:50:24 -0700987func NewCCLibrary(library *CCLibrary, module CCModuleType,
988 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
989
Colin Crossfa138792015-04-24 17:31:52 -0700990 return newCCDynamic(&library.CCLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -0700991 &library.LibraryProperties)
992}
993
994func CCLibraryFactory() (blueprint.Module, []interface{}) {
995 module := &CCLibrary{}
996
997 module.LibraryProperties.BuildShared = true
998 module.LibraryProperties.BuildStatic = true
999
1000 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
1001}
1002
Colin Cross0676e2d2015-04-24 17:39:18 -07001003func (c *CCLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001004 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Cross2732e9a2015-04-28 13:23:52 -07001005 if c.static() {
1006 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Static.Whole_static_libs...)
1007 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Static.Static_libs...)
1008 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Static.Shared_libs...)
1009 } else {
Colin Crossf6566ed2015-03-24 11:13:38 -07001010 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001011 if c.Properties.Sdk_version == "" {
1012 depNames.CrtBegin = "crtbegin_so"
1013 depNames.CrtEnd = "crtend_so"
1014 } else {
1015 depNames.CrtBegin = "ndk_crtbegin_so." + c.Properties.Sdk_version
1016 depNames.CrtEnd = "ndk_crtend_so." + c.Properties.Sdk_version
1017 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001018 }
Colin Cross2732e9a2015-04-28 13:23:52 -07001019 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Shared.Whole_static_libs...)
1020 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Shared.Static_libs...)
1021 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Shared.Shared_libs...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001022 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001023
Colin Cross21b9a242015-03-24 14:15:58 -07001024 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001025}
1026
Colin Cross97ba0732015-03-23 17:50:24 -07001027func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001028 return c.out
1029}
1030
Colin Crossed4cf0b2015-03-26 14:43:45 -07001031func (c *CCLibrary) getReuseObjFiles() []string {
1032 return c.reuseObjFiles
1033}
1034
1035func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
1036 c.reuseFrom = reuseFrom
1037}
1038
1039func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
1040 return c.reuseFrom
1041}
1042
Colin Cross97ba0732015-03-23 17:50:24 -07001043func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001044 return c.objFiles
1045}
1046
Colin Cross28344522015-04-22 13:07:53 -07001047func (c *CCLibrary) exportedFlags() []string {
1048 return c.exportFlags
Colin Cross3f40fa42015-01-30 17:27:36 -08001049}
1050
Colin Cross0676e2d2015-04-24 17:39:18 -07001051func (c *CCLibrary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001052 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001053
Colin Cross97ba0732015-03-23 17:50:24 -07001054 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -08001055
Colin Crossd8e780d2015-04-28 17:39:43 -07001056 if c.static() {
1057 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Static.Cflags...)
1058 } else {
1059 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Shared.Cflags...)
1060 }
1061
Colin Cross18b6dc52015-04-28 13:20:37 -07001062 if !c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001063 libName := ctx.ModuleName()
1064 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
1065 sharedFlag := "-Wl,-shared"
Colin Crossfa138792015-04-24 17:31:52 -07001066 if c.Properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001067 sharedFlag = "-shared"
1068 }
Colin Crossf6566ed2015-03-24 11:13:38 -07001069 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001070 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001071 }
Colin Cross97ba0732015-03-23 17:50:24 -07001072
Colin Cross0af4b842015-04-30 16:36:18 -07001073 if ctx.Darwin() {
1074 flags.LdFlags = append(flags.LdFlags,
1075 "-dynamiclib",
1076 "-single_module",
1077 //"-read_only_relocs suppress",
1078 "-install_name @rpath/"+libName+sharedLibraryExtension,
1079 )
1080 } else {
1081 flags.LdFlags = append(flags.LdFlags,
1082 "-Wl,--gc-sections",
1083 sharedFlag,
1084 "-Wl,-soname,"+libName+sharedLibraryExtension,
1085 )
1086 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001087 }
Colin Cross97ba0732015-03-23 17:50:24 -07001088
1089 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001090}
1091
Colin Cross97ba0732015-03-23 17:50:24 -07001092func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1093 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001094
1095 staticFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001096 objFilesStatic := c.customCompileObjs(ctx, staticFlags, common.DeviceStaticLibrary,
Dan Willemsen2ef08f42015-06-30 18:15:24 -07001097 c.LibraryProperties.Static.Srcs, c.LibraryProperties.Static.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001098
1099 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001100 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001101
1102 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1103
Colin Cross0af4b842015-04-30 16:36:18 -07001104 if ctx.Darwin() {
1105 TransformDarwinObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1106 } else {
1107 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1108 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001109
1110 c.objFiles = objFiles
1111 c.out = outputFile
Colin Crossf2298272015-05-12 11:36:53 -07001112
1113 common.CheckModuleSrcDirsExist(ctx, c.Properties.Export_include_dirs, "export_include_dirs")
Colin Crossfa138792015-04-24 17:31:52 -07001114 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001115 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001116
1117 ctx.CheckbuildFile(outputFile)
1118}
1119
Colin Cross97ba0732015-03-23 17:50:24 -07001120func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1121 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001122
1123 sharedFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001124 objFilesShared := c.customCompileObjs(ctx, sharedFlags, common.DeviceSharedLibrary,
Dan Willemsen2ef08f42015-06-30 18:15:24 -07001125 c.LibraryProperties.Shared.Srcs, c.LibraryProperties.Shared.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001126
1127 objFiles = append(objFiles, objFilesShared...)
1128
1129 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1130
Colin Cross97ba0732015-03-23 17:50:24 -07001131 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001132 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, false,
Colin Cross77b00fa2015-03-16 16:15:49 -07001133 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001134
1135 c.out = outputFile
Colin Crossfa138792015-04-24 17:31:52 -07001136 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001137 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001138}
1139
Colin Cross97ba0732015-03-23 17:50:24 -07001140func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1141 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001142
1143 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001144 if c.getReuseFrom().ccLibrary() == c {
1145 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001146 } else {
Colin Cross2732e9a2015-04-28 13:23:52 -07001147 if c.getReuseFrom().ccLibrary().LibraryProperties.Static.Cflags == nil &&
1148 c.LibraryProperties.Shared.Cflags == nil {
1149 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
1150 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001151 }
1152
Colin Crossed4cf0b2015-03-26 14:43:45 -07001153 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001154 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1155 } else {
1156 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1157 }
1158}
1159
Colin Cross97ba0732015-03-23 17:50:24 -07001160func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001161 // Static libraries do not get installed.
1162}
1163
Colin Cross97ba0732015-03-23 17:50:24 -07001164func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001165 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001166 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001167 installDir = "lib64"
1168 }
1169
Colin Crossfa138792015-04-24 17:31:52 -07001170 ctx.InstallFile(filepath.Join(installDir, c.Properties.Relative_install_path), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001171}
1172
Colin Cross97ba0732015-03-23 17:50:24 -07001173func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001174 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001175 c.installStaticLibrary(ctx, flags)
1176 } else {
1177 c.installSharedLibrary(ctx, flags)
1178 }
1179}
1180
Colin Cross3f40fa42015-01-30 17:27:36 -08001181//
1182// Objects (for crt*.o)
1183//
1184
Dan Albertc3144b12015-04-28 18:17:56 -07001185type ccObjectProvider interface {
1186 object() *ccObject
1187}
1188
Colin Cross3f40fa42015-01-30 17:27:36 -08001189type ccObject struct {
Colin Crossfa138792015-04-24 17:31:52 -07001190 CCBase
Colin Cross3f40fa42015-01-30 17:27:36 -08001191 out string
1192}
1193
Dan Albertc3144b12015-04-28 18:17:56 -07001194func (c *ccObject) object() *ccObject {
1195 return c
1196}
1197
Colin Cross97ba0732015-03-23 17:50:24 -07001198func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001199 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001200
Colin Crossfa138792015-04-24 17:31:52 -07001201 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001202}
1203
1204func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1205 // object files can't have any dynamic dependencies
1206 return nil
1207}
1208
Colin Cross0676e2d2015-04-24 17:39:18 -07001209func (*ccObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001210 // object files can't have any dynamic dependencies
1211 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001212}
1213
1214func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001215 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001216
Colin Cross97ba0732015-03-23 17:50:24 -07001217 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001218
1219 var outputFile string
1220 if len(objFiles) == 1 {
1221 outputFile = objFiles[0]
1222 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001223 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+objectExtension)
Colin Cross3f40fa42015-01-30 17:27:36 -08001224 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1225 }
1226
1227 c.out = outputFile
1228
1229 ctx.CheckbuildFile(outputFile)
1230}
1231
Colin Cross97ba0732015-03-23 17:50:24 -07001232func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001233 // Object files do not get installed.
1234}
1235
Colin Cross3f40fa42015-01-30 17:27:36 -08001236func (c *ccObject) outputFile() string {
1237 return c.out
1238}
1239
Dan Albertc3144b12015-04-28 18:17:56 -07001240var _ ccObjectProvider = (*ccObject)(nil)
1241
Colin Cross3f40fa42015-01-30 17:27:36 -08001242//
1243// Executables
1244//
1245
Colin Cross7d5136f2015-05-11 13:39:40 -07001246type CCBinaryProperties struct {
1247 // compile executable with -static
1248 Static_executable bool
1249
1250 // set the name of the output
1251 Stem string `android:"arch_variant"`
1252
1253 // append to the name of the output
1254 Suffix string `android:"arch_variant"`
1255
1256 // if set, add an extra objcopy --prefix-symbols= step
1257 Prefix_symbols string
1258}
1259
Colin Cross97ba0732015-03-23 17:50:24 -07001260type CCBinary struct {
Colin Crossfa138792015-04-24 17:31:52 -07001261 CCLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001262 out string
Colin Crossd350ecd2015-04-28 13:25:36 -07001263 installFile string
Colin Cross7d5136f2015-05-11 13:39:40 -07001264 BinaryProperties CCBinaryProperties
Colin Cross3f40fa42015-01-30 17:27:36 -08001265}
1266
Colin Crossed4cf0b2015-03-26 14:43:45 -07001267func (c *CCBinary) buildStatic() bool {
1268 return c.BinaryProperties.Static_executable
1269}
1270
1271func (c *CCBinary) buildShared() bool {
1272 return !c.BinaryProperties.Static_executable
1273}
1274
Colin Cross97ba0732015-03-23 17:50:24 -07001275func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001276 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001277 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001278 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001279 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001280
1281 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001282}
1283
Colin Cross0676e2d2015-04-24 17:39:18 -07001284func (c *CCBinary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001285 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001286 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001287 if c.Properties.Sdk_version == "" {
1288 if c.BinaryProperties.Static_executable {
1289 depNames.CrtBegin = "crtbegin_static"
1290 } else {
1291 depNames.CrtBegin = "crtbegin_dynamic"
1292 }
1293 depNames.CrtEnd = "crtend_android"
Colin Cross3f40fa42015-01-30 17:27:36 -08001294 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001295 if c.BinaryProperties.Static_executable {
1296 depNames.CrtBegin = "ndk_crtbegin_static." + c.Properties.Sdk_version
1297 } else {
1298 depNames.CrtBegin = "ndk_crtbegin_dynamic." + c.Properties.Sdk_version
1299 }
1300 depNames.CrtEnd = "ndk_crtend_android." + c.Properties.Sdk_version
Colin Cross3f40fa42015-01-30 17:27:36 -08001301 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001302
1303 if c.BinaryProperties.Static_executable {
Colin Cross74d1ec02015-04-28 13:30:13 -07001304 if c.stl(ctx) == "libc++_static" {
1305 depNames.StaticLibs = append(depNames.StaticLibs, "libm", "libc", "libdl")
1306 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001307 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1308 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1309 // move them to the beginning of deps.LateStaticLibs
1310 var groupLibs []string
1311 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1312 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1313 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1314 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001315 }
Colin Cross21b9a242015-03-24 14:15:58 -07001316 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001317}
1318
Colin Cross97ba0732015-03-23 17:50:24 -07001319func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001320 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001321
Colin Cross1f8f2342015-03-26 16:09:47 -07001322 props = append(props, &binary.BinaryProperties)
1323
Colin Crossfa138792015-04-24 17:31:52 -07001324 return newCCDynamic(&binary.CCLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001325}
1326
Colin Cross97ba0732015-03-23 17:50:24 -07001327func CCBinaryFactory() (blueprint.Module, []interface{}) {
1328 module := &CCBinary{}
1329
1330 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001331}
1332
Colin Cross18b6dc52015-04-28 13:20:37 -07001333func (c *CCBinary) ModifyProperties(ctx common.AndroidBaseContext) {
Colin Cross0af4b842015-04-30 16:36:18 -07001334 if ctx.Darwin() {
1335 c.BinaryProperties.Static_executable = false
1336 }
Colin Cross18b6dc52015-04-28 13:20:37 -07001337 if c.BinaryProperties.Static_executable {
1338 c.dynamicProperties.VariantIsStaticBinary = true
1339 }
1340}
1341
Colin Cross0676e2d2015-04-24 17:39:18 -07001342func (c *CCBinary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001343 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001344
Colin Cross97ba0732015-03-23 17:50:24 -07001345 flags.CFlags = append(flags.CFlags, "-fpie")
1346
Colin Crossf6566ed2015-03-24 11:13:38 -07001347 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001348 if c.BinaryProperties.Static_executable {
1349 // Clang driver needs -static to create static executable.
1350 // However, bionic/linker uses -shared to overwrite.
1351 // Linker for x86 targets does not allow coexistance of -static and -shared,
1352 // so we add -static only if -shared is not used.
1353 if !inList("-shared", flags.LdFlags) {
1354 flags.LdFlags = append(flags.LdFlags, "-static")
1355 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001356
Colin Crossed4cf0b2015-03-26 14:43:45 -07001357 flags.LdFlags = append(flags.LdFlags,
1358 "-nostdlib",
1359 "-Bstatic",
1360 "-Wl,--gc-sections",
1361 )
1362
1363 } else {
1364 linker := "/system/bin/linker"
1365 if flags.Toolchain.Is64Bit() {
1366 linker = "/system/bin/linker64"
1367 }
1368
1369 flags.LdFlags = append(flags.LdFlags,
1370 "-nostdlib",
1371 "-Bdynamic",
1372 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1373 "-Wl,--gc-sections",
1374 "-Wl,-z,nocopyreloc",
1375 )
1376 }
Colin Cross0af4b842015-04-30 16:36:18 -07001377 } else if ctx.Darwin() {
1378 flags.LdFlags = append(flags.LdFlags, "-Wl,-headerpad_max_install_names")
Colin Cross3f40fa42015-01-30 17:27:36 -08001379 }
1380
Colin Cross97ba0732015-03-23 17:50:24 -07001381 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001382}
1383
Colin Cross97ba0732015-03-23 17:50:24 -07001384func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1385 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001386
Colin Crossfa138792015-04-24 17:31:52 -07001387 if !c.BinaryProperties.Static_executable && inList("libc", c.Properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001388 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1389 "from static libs or set static_executable: true")
1390 }
1391
1392 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001393 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001394 if c.BinaryProperties.Prefix_symbols != "" {
1395 afterPrefixSymbols := outputFile
1396 outputFile = outputFile + ".intermediate"
1397 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1398 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1399 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001400
Colin Cross97ba0732015-03-23 17:50:24 -07001401 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossed4cf0b2015-03-26 14:43:45 -07001402 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001403 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001404}
Colin Cross3f40fa42015-01-30 17:27:36 -08001405
Colin Cross97ba0732015-03-23 17:50:24 -07001406func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossd350ecd2015-04-28 13:25:36 -07001407 c.installFile = ctx.InstallFile(filepath.Join("bin", c.Properties.Relative_install_path), c.out)
1408}
1409
1410func (c *CCBinary) HostToolPath() string {
1411 if c.HostOrDevice().Host() {
1412 return c.installFile
1413 }
1414 return ""
Dan Albertc403f7c2015-03-18 14:01:18 -07001415}
1416
Colin Cross7d5136f2015-05-11 13:39:40 -07001417type CCTestProperties struct {
1418 // Create a separate test for each source file. Useful when there is
1419 // global state that can not be torn down and reset between each test suite.
1420 Test_per_src bool
1421}
1422
Colin Cross9ffb4f52015-04-24 17:48:09 -07001423type CCTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001424 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001425
Colin Cross7d5136f2015-05-11 13:39:40 -07001426 TestProperties CCTestProperties
Dan Albertc403f7c2015-03-18 14:01:18 -07001427}
1428
Colin Cross9ffb4f52015-04-24 17:48:09 -07001429func (c *CCTest) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross0676e2d2015-04-24 17:39:18 -07001430 flags = c.CCBinary.flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001431
Colin Cross97ba0732015-03-23 17:50:24 -07001432 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001433 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001434 flags.CFlags = append(flags.CFlags, "-O0", "-g")
Colin Cross28344522015-04-22 13:07:53 -07001435 flags.LdFlags = append(flags.LdFlags, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001436 }
1437
1438 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross28344522015-04-22 13:07:53 -07001439 flags.CFlags = append(flags.CFlags,
1440 "-I"+filepath.Join(ctx.AConfig().SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001441
Colin Cross21b9a242015-03-24 14:15:58 -07001442 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001443}
1444
Colin Cross9ffb4f52015-04-24 17:48:09 -07001445func (c *CCTest) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001446 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
Colin Crossa8a93d32015-04-28 13:26:49 -07001447 depNames = c.CCBinary.depNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -07001448 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001449}
1450
Colin Cross9ffb4f52015-04-24 17:48:09 -07001451func (c *CCTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001452 if ctx.Device() {
Colin Crossa8a93d32015-04-28 13:26:49 -07001453 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001454 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001455 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001456 }
1457}
1458
Colin Cross9ffb4f52015-04-24 17:48:09 -07001459func (c *CCTest) testPerSrc() bool {
1460 return c.TestProperties.Test_per_src
Colin Cross6b290692015-03-19 14:05:33 -07001461}
1462
Colin Cross9ffb4f52015-04-24 17:48:09 -07001463func (c *CCTest) test() *CCTest {
1464 return c
1465}
1466
1467func NewCCTest(test *CCTest, module CCModuleType,
1468 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1469
1470 props = append(props, &test.TestProperties)
1471
1472 return NewCCBinary(&test.CCBinary, module, hod, props...)
1473}
1474
1475func CCTestFactory() (blueprint.Module, []interface{}) {
1476 module := &CCTest{}
1477
1478 return NewCCTest(module, module, common.HostAndDeviceSupported)
1479}
1480
1481type testPerSrc interface {
1482 test() *CCTest
1483 testPerSrc() bool
1484}
1485
1486var _ testPerSrc = (*CCTest)(nil)
1487
Colin Cross6b290692015-03-19 14:05:33 -07001488func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001489 if test, ok := mctx.Module().(testPerSrc); ok {
1490 if test.testPerSrc() {
1491 testNames := make([]string, len(test.test().Properties.Srcs))
1492 for i, src := range test.test().Properties.Srcs {
Colin Cross6b290692015-03-19 14:05:33 -07001493 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1494 }
1495 tests := mctx.CreateLocalVariations(testNames...)
Colin Cross9ffb4f52015-04-24 17:48:09 -07001496 for i, src := range test.test().Properties.Srcs {
1497 tests[i].(testPerSrc).test().Properties.Srcs = []string{src}
1498 tests[i].(testPerSrc).test().BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001499 }
1500 }
1501 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001502}
1503
Colin Cross2ba19d92015-05-07 15:44:20 -07001504type CCBenchmark struct {
1505 CCBinary
1506}
1507
1508func (c *CCBenchmark) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1509 depNames = c.CCBinary.depNames(ctx, depNames)
1510 depNames.StaticLibs = append(depNames.StaticLibs, "libbenchmark")
1511 return depNames
1512}
1513
1514func (c *CCBenchmark) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1515 if ctx.Device() {
1516 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
1517 } else {
1518 c.CCBinary.installModule(ctx, flags)
1519 }
1520}
1521
1522func NewCCBenchmark(test *CCBenchmark, module CCModuleType,
1523 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1524
1525 return NewCCBinary(&test.CCBinary, module, hod, props...)
1526}
1527
1528func CCBenchmarkFactory() (blueprint.Module, []interface{}) {
1529 module := &CCBenchmark{}
1530
1531 return NewCCBenchmark(module, module, common.HostAndDeviceSupported)
1532}
1533
Colin Cross3f40fa42015-01-30 17:27:36 -08001534//
1535// Static library
1536//
1537
Colin Cross97ba0732015-03-23 17:50:24 -07001538func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1539 module := &CCLibrary{}
1540 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001541
Colin Cross97ba0732015-03-23 17:50:24 -07001542 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001543}
1544
1545//
1546// Shared libraries
1547//
1548
Colin Cross97ba0732015-03-23 17:50:24 -07001549func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1550 module := &CCLibrary{}
1551 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001552
Colin Cross97ba0732015-03-23 17:50:24 -07001553 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001554}
1555
1556//
1557// Host static library
1558//
1559
Colin Cross97ba0732015-03-23 17:50:24 -07001560func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1561 module := &CCLibrary{}
1562 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001563
Colin Cross97ba0732015-03-23 17:50:24 -07001564 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001565}
1566
1567//
1568// Host Shared libraries
1569//
1570
Colin Cross97ba0732015-03-23 17:50:24 -07001571func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1572 module := &CCLibrary{}
1573 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001574
Colin Cross97ba0732015-03-23 17:50:24 -07001575 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001576}
1577
1578//
1579// Host Binaries
1580//
1581
Colin Cross97ba0732015-03-23 17:50:24 -07001582func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1583 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001584
Colin Cross97ba0732015-03-23 17:50:24 -07001585 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001586}
1587
1588//
Colin Cross1f8f2342015-03-26 16:09:47 -07001589// Host Tests
1590//
1591
1592func CCTestHostFactory() (blueprint.Module, []interface{}) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001593 module := &CCTest{}
Colin Cross1f8f2342015-03-26 16:09:47 -07001594 return NewCCBinary(&module.CCBinary, module, common.HostSupported,
Colin Cross9ffb4f52015-04-24 17:48:09 -07001595 &module.TestProperties)
Colin Cross1f8f2342015-03-26 16:09:47 -07001596}
1597
1598//
Colin Cross2ba19d92015-05-07 15:44:20 -07001599// Host Benchmarks
1600//
1601
1602func CCBenchmarkHostFactory() (blueprint.Module, []interface{}) {
1603 module := &CCBenchmark{}
1604 return NewCCBinary(&module.CCBinary, module, common.HostSupported)
1605}
1606
1607//
Colin Cross3f40fa42015-01-30 17:27:36 -08001608// Device libraries shipped with gcc
1609//
1610
1611type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001612 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001613}
1614
1615func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1616 // toolchain libraries can't have any dependencies
1617 return nil
1618}
1619
Colin Cross0676e2d2015-04-24 17:39:18 -07001620func (*toolchainLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001621 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001622 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001623}
1624
Colin Cross97ba0732015-03-23 17:50:24 -07001625func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001626 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001627
Colin Cross97ba0732015-03-23 17:50:24 -07001628 module.LibraryProperties.BuildStatic = true
1629
Colin Crossfa138792015-04-24 17:31:52 -07001630 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth,
Colin Cross21b9a242015-03-24 14:15:58 -07001631 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001632}
1633
1634func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001635 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001636
1637 libName := ctx.ModuleName() + staticLibraryExtension
1638 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1639
1640 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1641
1642 c.out = outputFile
1643
1644 ctx.CheckbuildFile(outputFile)
1645}
1646
Colin Cross97ba0732015-03-23 17:50:24 -07001647func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001648 // Toolchain libraries do not get installed.
1649}
1650
Dan Albertbe961682015-03-18 23:38:50 -07001651// NDK prebuilt libraries.
1652//
1653// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1654// either (with the exception of the shared STLs, which are installed to the app's directory rather
1655// than to the system image).
1656
1657func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1658 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
Colin Cross1332b002015-04-07 17:11:30 -07001659 ctx.AConfig().SrcDir(), version, toolchain.Name())
Dan Albertbe961682015-03-18 23:38:50 -07001660}
1661
Dan Albertc3144b12015-04-28 18:17:56 -07001662func ndkPrebuiltModuleToPath(ctx common.AndroidModuleContext, toolchain Toolchain,
1663 ext string, version string) string {
1664
1665 // NDK prebuilts are named like: ndk_NAME.EXT.SDK_VERSION.
1666 // We want to translate to just NAME.EXT
1667 name := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1668 dir := getNdkLibDir(ctx, toolchain, version)
1669 return filepath.Join(dir, name+ext)
1670}
1671
1672type ndkPrebuiltObject struct {
1673 ccObject
1674}
1675
1676func (*ndkPrebuiltObject) AndroidDynamicDependencies(
1677 ctx common.AndroidDynamicDependerModuleContext) []string {
1678
1679 // NDK objects can't have any dependencies
1680 return nil
1681}
1682
1683func (*ndkPrebuiltObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1684 // NDK objects can't have any dependencies
1685 return CCDeps{}
1686}
1687
1688func NdkPrebuiltObjectFactory() (blueprint.Module, []interface{}) {
1689 module := &ndkPrebuiltObject{}
1690 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
1691}
1692
1693func (c *ndkPrebuiltObject) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1694 deps CCDeps, objFiles []string) {
1695 // A null build step, but it sets up the output path.
1696 if !strings.HasPrefix(ctx.ModuleName(), "ndk_crt") {
1697 ctx.ModuleErrorf("NDK prebuilts must have an ndk_crt prefixed name")
1698 }
1699
1700 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, objectExtension, c.Properties.Sdk_version)
1701}
1702
1703func (c *ndkPrebuiltObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1704 // Objects do not get installed.
1705}
1706
1707var _ ccObjectProvider = (*ndkPrebuiltObject)(nil)
1708
Dan Albertbe961682015-03-18 23:38:50 -07001709type ndkPrebuiltLibrary struct {
1710 CCLibrary
1711}
1712
1713func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1714 ctx common.AndroidDynamicDependerModuleContext) []string {
1715
1716 // NDK libraries can't have any dependencies
1717 return nil
1718}
1719
Colin Cross0676e2d2015-04-24 17:39:18 -07001720func (*ndkPrebuiltLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Dan Albertbe961682015-03-18 23:38:50 -07001721 // NDK libraries can't have any dependencies
1722 return CCDeps{}
1723}
1724
1725func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1726 module := &ndkPrebuiltLibrary{}
1727 module.LibraryProperties.BuildShared = true
1728 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1729}
1730
1731func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1732 deps CCDeps, objFiles []string) {
1733 // A null build step, but it sets up the output path.
1734 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1735 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1736 }
1737
Colin Crossfa138792015-04-24 17:31:52 -07001738 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001739 c.exportFlags = []string{common.JoinWithPrefix(includeDirs, "-isystem ")}
Dan Albertbe961682015-03-18 23:38:50 -07001740
Dan Albertc3144b12015-04-28 18:17:56 -07001741 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, sharedLibraryExtension,
1742 c.Properties.Sdk_version)
Dan Albertbe961682015-03-18 23:38:50 -07001743}
1744
1745func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc3144b12015-04-28 18:17:56 -07001746 // NDK prebuilt libraries do not get installed.
Dan Albertbe961682015-03-18 23:38:50 -07001747}
1748
1749// The NDK STLs are slightly different from the prebuilt system libraries:
1750// * Are not specific to each platform version.
1751// * The libraries are not in a predictable location for each STL.
1752
1753type ndkPrebuiltStl struct {
1754 ndkPrebuiltLibrary
1755}
1756
1757type ndkPrebuiltStaticStl struct {
1758 ndkPrebuiltStl
1759}
1760
1761type ndkPrebuiltSharedStl struct {
1762 ndkPrebuiltStl
1763}
1764
1765func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1766 module := &ndkPrebuiltSharedStl{}
1767 module.LibraryProperties.BuildShared = true
1768 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1769}
1770
1771func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1772 module := &ndkPrebuiltStaticStl{}
1773 module.LibraryProperties.BuildStatic = true
1774 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1775}
1776
1777func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1778 gccVersion := toolchain.GccVersion()
1779 var libDir string
1780 switch stl {
1781 case "libstlport":
1782 libDir = "cxx-stl/stlport/libs"
1783 case "libc++":
1784 libDir = "cxx-stl/llvm-libc++/libs"
1785 case "libgnustl":
1786 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1787 }
1788
1789 if libDir != "" {
Colin Cross1332b002015-04-07 17:11:30 -07001790 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources"
Dan Albertbe961682015-03-18 23:38:50 -07001791 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1792 }
1793
1794 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1795 return ""
1796}
1797
1798func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1799 deps CCDeps, objFiles []string) {
1800 // A null build step, but it sets up the output path.
1801 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1802 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1803 }
1804
Colin Crossfa138792015-04-24 17:31:52 -07001805 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001806 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Dan Albertbe961682015-03-18 23:38:50 -07001807
1808 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1809 libExt := sharedLibraryExtension
1810 if c.LibraryProperties.BuildStatic {
1811 libExt = staticLibraryExtension
1812 }
1813
1814 stlName := strings.TrimSuffix(libName, "_shared")
1815 stlName = strings.TrimSuffix(stlName, "_static")
1816 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1817 c.out = libDir + "/" + libName + libExt
1818}
1819
Colin Cross3f40fa42015-01-30 17:27:36 -08001820func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001821 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001822 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001823 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001824 modules = mctx.CreateLocalVariations("static", "shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001825 modules[0].(ccLinkedInterface).setStatic(true)
1826 modules[1].(ccLinkedInterface).setStatic(false)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001827 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001828 modules = mctx.CreateLocalVariations("static")
Colin Cross18b6dc52015-04-28 13:20:37 -07001829 modules[0].(ccLinkedInterface).setStatic(true)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001830 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001831 modules = mctx.CreateLocalVariations("shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001832 modules[0].(ccLinkedInterface).setStatic(false)
Colin Cross3f40fa42015-01-30 17:27:36 -08001833 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001834 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001835 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001836
1837 if _, ok := c.(ccLibraryInterface); ok {
1838 reuseFrom := modules[0].(ccLibraryInterface)
1839 for _, m := range modules {
1840 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001841 }
1842 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001843 }
1844}
Colin Cross74d1ec02015-04-28 13:30:13 -07001845
1846// lastUniqueElements returns all unique elements of a slice, keeping the last copy of each
1847// modifies the slice contents in place, and returns a subslice of the original slice
1848func lastUniqueElements(list []string) []string {
1849 totalSkip := 0
1850 for i := len(list) - 1; i >= totalSkip; i-- {
1851 skip := 0
1852 for j := i - 1; j >= totalSkip; j-- {
1853 if list[i] == list[j] {
1854 skip++
1855 } else {
1856 list[j+skip] = list[j]
1857 }
1858 }
1859 totalSkip += skip
1860 }
1861 return list[totalSkip:]
1862}