blob: e9d3884181f08e5646e3597b6cb05a922f6c9054 [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 Cross463a90e2015-06-17 14:20:06 -070030 "android/soong"
Colin Cross3f40fa42015-01-30 17:27:36 -080031 "android/soong/common"
Colin Cross5049f022015-03-18 13:28:46 -070032 "android/soong/genrule"
Colin Cross3f40fa42015-01-30 17:27:36 -080033)
34
Colin Cross463a90e2015-06-17 14:20:06 -070035func init() {
36 soong.RegisterModuleType("cc_library_static", CCLibraryStaticFactory)
37 soong.RegisterModuleType("cc_library_shared", CCLibrarySharedFactory)
38 soong.RegisterModuleType("cc_library", CCLibraryFactory)
39 soong.RegisterModuleType("cc_object", CCObjectFactory)
40 soong.RegisterModuleType("cc_binary", CCBinaryFactory)
41 soong.RegisterModuleType("cc_test", CCTestFactory)
42 soong.RegisterModuleType("cc_benchmark", CCBenchmarkFactory)
43
44 soong.RegisterModuleType("toolchain_library", ToolchainLibraryFactory)
45 soong.RegisterModuleType("ndk_prebuilt_library", NdkPrebuiltLibraryFactory)
46 soong.RegisterModuleType("ndk_prebuilt_object", NdkPrebuiltObjectFactory)
47 soong.RegisterModuleType("ndk_prebuilt_static_stl", NdkPrebuiltStaticStlFactory)
48 soong.RegisterModuleType("ndk_prebuilt_shared_stl", NdkPrebuiltSharedStlFactory)
49
50 soong.RegisterModuleType("cc_library_host_static", CCLibraryHostStaticFactory)
51 soong.RegisterModuleType("cc_library_host_shared", CCLibraryHostSharedFactory)
52 soong.RegisterModuleType("cc_binary_host", CCBinaryHostFactory)
53 soong.RegisterModuleType("cc_test_host", CCTestHostFactory)
54 soong.RegisterModuleType("cc_benchmark_host", CCBenchmarkHostFactory)
55
56 // LinkageMutator must be registered after common.ArchMutator, but that is guaranteed by
57 // the Go initialization order because this package depends on common, so common's init
58 // functions will run first.
59 soong.RegisterEarlyMutator("link", LinkageMutator)
60 soong.RegisterEarlyMutator("test_per_src", TestPerSrcMutator)
61}
62
Colin Cross3f40fa42015-01-30 17:27:36 -080063var (
Colin Cross1332b002015-04-07 17:11:30 -070064 HostPrebuiltTag = pctx.VariableConfigMethod("HostPrebuiltTag", common.Config.PrebuiltOS)
65 SrcDir = pctx.VariableConfigMethod("SrcDir", common.Config.SrcDir)
Colin Cross3f40fa42015-01-30 17:27:36 -080066
67 LibcRoot = pctx.StaticVariable("LibcRoot", "${SrcDir}/bionic/libc")
68 LibmRoot = pctx.StaticVariable("LibmRoot", "${SrcDir}/bionic/libm")
69)
70
71// Flags used by lots of devices. Putting them in package static variables will save bytes in
72// build.ninja so they aren't repeated for every file
73var (
74 commonGlobalCflags = []string{
75 "-DANDROID",
76 "-fmessage-length=0",
77 "-W",
78 "-Wall",
79 "-Wno-unused",
80 "-Winit-self",
81 "-Wpointer-arith",
82
83 // COMMON_RELEASE_CFLAGS
84 "-DNDEBUG",
85 "-UDEBUG",
86 }
87
88 deviceGlobalCflags = []string{
89 // TARGET_ERROR_FLAGS
90 "-Werror=return-type",
91 "-Werror=non-virtual-dtor",
92 "-Werror=address",
93 "-Werror=sequence-point",
94 }
95
96 hostGlobalCflags = []string{}
97
98 commonGlobalCppflags = []string{
99 "-Wsign-promo",
100 "-std=gnu++11",
101 }
102)
103
104func init() {
105 pctx.StaticVariable("commonGlobalCflags", strings.Join(commonGlobalCflags, " "))
106 pctx.StaticVariable("deviceGlobalCflags", strings.Join(deviceGlobalCflags, " "))
107 pctx.StaticVariable("hostGlobalCflags", strings.Join(hostGlobalCflags, " "))
108
109 pctx.StaticVariable("commonGlobalCppflags", strings.Join(commonGlobalCppflags, " "))
110
111 pctx.StaticVariable("commonClangGlobalCflags",
112 strings.Join(clangFilterUnknownCflags(commonGlobalCflags), " "))
113 pctx.StaticVariable("deviceClangGlobalCflags",
114 strings.Join(clangFilterUnknownCflags(deviceGlobalCflags), " "))
115 pctx.StaticVariable("hostClangGlobalCflags",
116 strings.Join(clangFilterUnknownCflags(hostGlobalCflags), " "))
Tim Kilbournf2948142015-03-11 12:03:03 -0700117 pctx.StaticVariable("commonClangGlobalCppflags",
118 strings.Join(clangFilterUnknownCflags(commonGlobalCppflags), " "))
Colin Cross3f40fa42015-01-30 17:27:36 -0800119
120 // Everything in this list is a crime against abstraction and dependency tracking.
121 // Do not add anything to this list.
122 pctx.StaticVariable("commonGlobalIncludes", strings.Join([]string{
123 "-isystem ${SrcDir}/system/core/include",
124 "-isystem ${SrcDir}/hardware/libhardware/include",
125 "-isystem ${SrcDir}/hardware/libhardware_legacy/include",
126 "-isystem ${SrcDir}/hardware/ril/include",
127 "-isystem ${SrcDir}/libnativehelper/include",
128 "-isystem ${SrcDir}/frameworks/native/include",
129 "-isystem ${SrcDir}/frameworks/native/opengl/include",
130 "-isystem ${SrcDir}/frameworks/av/include",
131 "-isystem ${SrcDir}/frameworks/base/include",
132 }, " "))
133
134 pctx.StaticVariable("clangPath", "${SrcDir}/prebuilts/clang/${HostPrebuiltTag}/host/3.6/bin/")
135}
136
Colin Cross3f40fa42015-01-30 17:27:36 -0800137// Building C/C++ code is handled by objects that satisfy this interface via composition
Colin Cross97ba0732015-03-23 17:50:24 -0700138type CCModuleType interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800139 common.AndroidModule
140
Colin Crossfa138792015-04-24 17:31:52 -0700141 // Modify property values after parsing Blueprints file but before starting dependency
142 // resolution or build rule generation
143 ModifyProperties(common.AndroidBaseContext)
144
Colin Cross21b9a242015-03-24 14:15:58 -0700145 // Modify the ccFlags
Colin Cross0676e2d2015-04-24 17:39:18 -0700146 flags(common.AndroidModuleContext, CCFlags) CCFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800147
Colin Cross21b9a242015-03-24 14:15:58 -0700148 // Return list of dependency names for use in AndroidDynamicDependencies and in depsToPaths
Colin Cross0676e2d2015-04-24 17:39:18 -0700149 depNames(common.AndroidBaseContext, CCDeps) CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800150
151 // Compile objects into final module
Colin Cross97ba0732015-03-23 17:50:24 -0700152 compileModule(common.AndroidModuleContext, CCFlags, CCDeps, []string)
Colin Cross3f40fa42015-01-30 17:27:36 -0800153
Dan Albertc403f7c2015-03-18 14:01:18 -0700154 // Install the built module.
Colin Cross97ba0732015-03-23 17:50:24 -0700155 installModule(common.AndroidModuleContext, CCFlags)
Dan Albertc403f7c2015-03-18 14:01:18 -0700156
Colin Cross3f40fa42015-01-30 17:27:36 -0800157 // Return the output file (.o, .a or .so) for use by other modules
158 outputFile() string
159}
160
Colin Cross97ba0732015-03-23 17:50:24 -0700161type CCDeps struct {
Colin Cross28344522015-04-22 13:07:53 -0700162 StaticLibs, SharedLibs, LateStaticLibs, WholeStaticLibs, ObjFiles, Cflags []string
Colin Crossc472d572015-03-17 15:06:21 -0700163
Colin Cross21b9a242015-03-24 14:15:58 -0700164 WholeStaticLibObjFiles []string
165
Colin Cross97ba0732015-03-23 17:50:24 -0700166 CrtBegin, CrtEnd string
Colin Crossc472d572015-03-17 15:06:21 -0700167}
168
Colin Cross97ba0732015-03-23 17:50:24 -0700169type CCFlags struct {
Colin Cross28344522015-04-22 13:07:53 -0700170 GlobalFlags []string // Flags that apply to C, C++, and assembly source files
171 AsFlags []string // Flags that apply to assembly source files
172 CFlags []string // Flags that apply to C and C++ source files
173 ConlyFlags []string // Flags that apply to C source files
174 CppFlags []string // Flags that apply to C++ source files
175 YaccFlags []string // Flags that apply to Yacc source files
176 LdFlags []string // Flags that apply to linker command lines
177
178 Nocrt bool
179 Toolchain Toolchain
180 Clang bool
Colin Crossc472d572015-03-17 15:06:21 -0700181}
182
Colin Cross7d5136f2015-05-11 13:39:40 -0700183// Properties used to compile all C or C++ modules
184type CCBaseProperties struct {
185 // 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 -0700186 Srcs []string `android:"arch_variant"`
187
188 // list of source files that should not be used to build the C/C++ module.
189 // This is most useful in the arch/multilib variants to remove non-common files
190 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700191
192 // list of module-specific flags that will be used for C and C++ compiles.
193 Cflags []string `android:"arch_variant"`
194
195 // list of module-specific flags that will be used for C++ compiles
196 Cppflags []string `android:"arch_variant"`
197
198 // list of module-specific flags that will be used for C compiles
199 Conlyflags []string `android:"arch_variant"`
200
201 // list of module-specific flags that will be used for .S compiles
202 Asflags []string `android:"arch_variant"`
203
204 // list of module-specific flags that will be used for .y and .yy compiles
205 Yaccflags []string
206
207 // list of module-specific flags that will be used for all link steps
208 Ldflags []string `android:"arch_variant"`
209
210 // the instruction set architecture to use to compile the C/C++
211 // module.
212 Instruction_set string `android:"arch_variant"`
213
214 // list of directories relative to the root of the source tree that will
215 // be added to the include path using -I.
216 // If possible, don't use this. If adding paths from the current directory use
217 // local_include_dirs, if adding paths from other modules use export_include_dirs in
218 // that module.
219 Include_dirs []string `android:"arch_variant"`
220
221 // list of directories relative to the Blueprints file that will
222 // be added to the include path using -I
223 Local_include_dirs []string `android:"arch_variant"`
224
225 // list of directories relative to the Blueprints file that will
226 // be added to the include path using -I for any module that links against this module
227 Export_include_dirs []string `android:"arch_variant"`
228
229 // list of module-specific flags that will be used for C and C++ compiles when
230 // compiling with clang
231 Clang_cflags []string `android:"arch_variant"`
232
233 // list of module-specific flags that will be used for .S compiles when
234 // compiling with clang
235 Clang_asflags []string `android:"arch_variant"`
236
237 // list of system libraries that will be dynamically linked to
238 // shared library and executable modules. If unset, generally defaults to libc
239 // and libm. Set to [] to prevent linking against libc and libm.
240 System_shared_libs []string
241
242 // list of modules whose object files should be linked into this module
243 // in their entirety. For static library modules, all of the .o files from the intermediate
244 // directory of the dependency will be linked into this modules .a file. For a shared library,
245 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
246 Whole_static_libs []string `android:"arch_variant"`
247
248 // list of modules that should be statically linked into this module.
249 Static_libs []string `android:"arch_variant"`
250
251 // list of modules that should be dynamically linked into this module.
252 Shared_libs []string `android:"arch_variant"`
253
254 // allow the module to contain undefined symbols. By default,
255 // modules cannot contain undefined symbols that are not satisified by their immediate
256 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
257 // This flag should only be necessary for compiling low-level libraries like libc.
258 Allow_undefined_symbols bool
259
260 // don't link in crt_begin and crt_end. This flag should only be necessary for
261 // compiling crt or libc.
262 Nocrt bool `android:"arch_variant"`
263
264 // don't insert default compiler flags into asflags, cflags,
265 // cppflags, conlyflags, ldflags, or include_dirs
266 No_default_compiler_flags bool
267
268 // compile module with clang instead of gcc
269 Clang bool `android:"arch_variant"`
270
271 // pass -frtti instead of -fno-rtti
272 Rtti bool
273
274 // -l arguments to pass to linker for host-provided shared libraries
275 Host_ldlibs []string `android:"arch_variant"`
276
277 // select the STL library to use. Possible values are "libc++", "libc++_static",
278 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
279 // default
280 Stl string
281
282 // Set for combined shared/static libraries to prevent compiling object files a second time
283 SkipCompileObjs bool `blueprint:"mutated"`
284
285 Debug, Release struct {
286 // list of module-specific flags that will be used for C and C++ compiles in debug or
287 // release builds
288 Cflags []string `android:"arch_variant"`
289 } `android:"arch_variant"`
290
291 // Minimum sdk version supported when compiling against the ndk
292 Sdk_version string
293
294 // install to a subdirectory of the default install path for the module
295 Relative_install_path string
296}
297
Colin Crossfa138792015-04-24 17:31:52 -0700298// CCBase contains the properties and members used by all C/C++ module types, and implements
Colin Crossc472d572015-03-17 15:06:21 -0700299// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
300// and uses a ccModuleType interface to that struct to create the build steps.
Colin Crossfa138792015-04-24 17:31:52 -0700301type CCBase struct {
Colin Crossc472d572015-03-17 15:06:21 -0700302 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700303 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700304
Colin Cross7d5136f2015-05-11 13:39:40 -0700305 Properties CCBaseProperties
Colin Crossfa138792015-04-24 17:31:52 -0700306
307 unused struct {
308 Asan bool
309 Native_coverage bool
310 Strip string
311 Tags []string
312 Required []string
313 }
Colin Crossc472d572015-03-17 15:06:21 -0700314
315 installPath string
Colin Cross74d1ec02015-04-28 13:30:13 -0700316
317 savedDepNames CCDeps
Colin Crossc472d572015-03-17 15:06:21 -0700318}
319
Colin Crossfa138792015-04-24 17:31:52 -0700320func newCCBase(base *CCBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700321 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
322
323 base.module = module
324
Colin Crossfa138792015-04-24 17:31:52 -0700325 props = append(props, &base.Properties, &base.unused)
Colin Crossc472d572015-03-17 15:06:21 -0700326
Colin Cross5049f022015-03-18 13:28:46 -0700327 return common.InitAndroidArchModule(module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700328}
329
Colin Crossfa138792015-04-24 17:31:52 -0700330func (c *CCBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800331 toolchain := c.findToolchain(ctx)
332 if ctx.Failed() {
333 return
334 }
335
Colin Cross21b9a242015-03-24 14:15:58 -0700336 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800337 if ctx.Failed() {
338 return
339 }
340
Colin Cross74d1ec02015-04-28 13:30:13 -0700341 deps := c.depsToPaths(ctx, c.savedDepNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800342 if ctx.Failed() {
343 return
344 }
345
Colin Cross28344522015-04-22 13:07:53 -0700346 flags.CFlags = append(flags.CFlags, deps.Cflags...)
Colin Crossed9f8682015-03-18 17:17:35 -0700347
Colin Cross581c1892015-04-07 16:50:10 -0700348 objFiles := c.compileObjs(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800349 if ctx.Failed() {
350 return
351 }
352
Colin Cross581c1892015-04-07 16:50:10 -0700353 generatedObjFiles := c.compileGeneratedObjs(ctx, flags)
Colin Cross5049f022015-03-18 13:28:46 -0700354 if ctx.Failed() {
355 return
356 }
357
358 objFiles = append(objFiles, generatedObjFiles...)
359
Colin Cross3f40fa42015-01-30 17:27:36 -0800360 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
361 if ctx.Failed() {
362 return
363 }
Dan Albertc403f7c2015-03-18 14:01:18 -0700364
365 c.ccModuleType().installModule(ctx, flags)
366 if ctx.Failed() {
367 return
368 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800369}
370
Colin Crossfa138792015-04-24 17:31:52 -0700371func (c *CCBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800372 return c.module
373}
374
Colin Crossfa138792015-04-24 17:31:52 -0700375var _ common.AndroidDynamicDepender = (*CCBase)(nil)
Colin Cross3f40fa42015-01-30 17:27:36 -0800376
Colin Crossfa138792015-04-24 17:31:52 -0700377func (c *CCBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800378 arch := ctx.Arch()
Colin Crossd3ba0392015-05-07 14:11:29 -0700379 hod := ctx.HostOrDevice()
380 factory := toolchainFactories[hod][arch.ArchType]
Colin Cross3f40fa42015-01-30 17:27:36 -0800381 if factory == nil {
382 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
Colin Crossd3ba0392015-05-07 14:11:29 -0700383 hod.String(), arch.String()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800384 }
385 return factory(arch.ArchVariant, arch.CpuVariant)
386}
387
Colin Crossfa138792015-04-24 17:31:52 -0700388func (c *CCBase) ModifyProperties(ctx common.AndroidBaseContext) {
389}
390
Colin Crosse11befc2015-04-27 17:49:17 -0700391func (c *CCBase) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossfa138792015-04-24 17:31:52 -0700392 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.Properties.Whole_static_libs...)
393 depNames.StaticLibs = append(depNames.StaticLibs, c.Properties.Static_libs...)
394 depNames.SharedLibs = append(depNames.SharedLibs, c.Properties.Shared_libs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700395
Colin Cross21b9a242015-03-24 14:15:58 -0700396 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800397}
398
Colin Crossfa138792015-04-24 17:31:52 -0700399func (c *CCBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
400 c.module.ModifyProperties(ctx)
401
Colin Cross74d1ec02015-04-28 13:30:13 -0700402 c.savedDepNames = c.module.depNames(ctx, CCDeps{})
403 c.savedDepNames.WholeStaticLibs = lastUniqueElements(c.savedDepNames.WholeStaticLibs)
404 c.savedDepNames.StaticLibs = lastUniqueElements(c.savedDepNames.StaticLibs)
405 c.savedDepNames.SharedLibs = lastUniqueElements(c.savedDepNames.SharedLibs)
406
407 staticLibs := c.savedDepNames.WholeStaticLibs
408 staticLibs = append(staticLibs, c.savedDepNames.StaticLibs...)
409 staticLibs = append(staticLibs, c.savedDepNames.LateStaticLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700410 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800411
Colin Cross74d1ec02015-04-28 13:30:13 -0700412 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, c.savedDepNames.SharedLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700413
Colin Cross74d1ec02015-04-28 13:30:13 -0700414 ret := append([]string(nil), c.savedDepNames.ObjFiles...)
415 if c.savedDepNames.CrtBegin != "" {
416 ret = append(ret, c.savedDepNames.CrtBegin)
Colin Cross21b9a242015-03-24 14:15:58 -0700417 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700418 if c.savedDepNames.CrtEnd != "" {
419 ret = append(ret, c.savedDepNames.CrtEnd)
Colin Cross21b9a242015-03-24 14:15:58 -0700420 }
421
422 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800423}
424
425// Create a ccFlags struct that collects the compile flags from global values,
426// per-target values, module type values, and per-module Blueprints properties
Colin Crossfa138792015-04-24 17:31:52 -0700427func (c *CCBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700428 flags := CCFlags{
Colin Crossfa138792015-04-24 17:31:52 -0700429 CFlags: c.Properties.Cflags,
430 CppFlags: c.Properties.Cppflags,
431 ConlyFlags: c.Properties.Conlyflags,
432 LdFlags: c.Properties.Ldflags,
433 AsFlags: c.Properties.Asflags,
434 YaccFlags: c.Properties.Yaccflags,
435 Nocrt: c.Properties.Nocrt,
Colin Cross97ba0732015-03-23 17:50:24 -0700436 Toolchain: toolchain,
Colin Crossfa138792015-04-24 17:31:52 -0700437 Clang: c.Properties.Clang,
Colin Cross3f40fa42015-01-30 17:27:36 -0800438 }
Colin Cross28344522015-04-22 13:07:53 -0700439
440 // Include dir cflags
Colin Crossf2298272015-05-12 11:36:53 -0700441 common.CheckSrcDirsExist(ctx, c.Properties.Include_dirs, "include_dirs")
442 common.CheckModuleSrcDirsExist(ctx, c.Properties.Local_include_dirs, "local_include_dirs")
443
Colin Crossfa138792015-04-24 17:31:52 -0700444 rootIncludeDirs := pathtools.PrefixPaths(c.Properties.Include_dirs, ctx.AConfig().SrcDir())
445 localIncludeDirs := pathtools.PrefixPaths(c.Properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -0700446 flags.GlobalFlags = append(flags.GlobalFlags,
447 includeDirsToFlags(rootIncludeDirs),
448 includeDirsToFlags(localIncludeDirs))
449
Colin Crossfa138792015-04-24 17:31:52 -0700450 if !c.Properties.No_default_compiler_flags {
451 if c.Properties.Sdk_version == "" || ctx.Host() {
Colin Cross28344522015-04-22 13:07:53 -0700452 flags.GlobalFlags = append(flags.GlobalFlags,
453 "${commonGlobalIncludes}",
454 toolchain.IncludeFlags(),
455 "-I${SrcDir}/libnativehelper/include/nativehelper")
456 }
457
458 flags.GlobalFlags = append(flags.GlobalFlags, []string{
459 "-I" + common.ModuleSrcDir(ctx),
460 "-I" + common.ModuleOutDir(ctx),
461 "-I" + common.ModuleGenDir(ctx),
462 }...)
463 }
464
Colin Crossfa138792015-04-24 17:31:52 -0700465 instructionSet := c.Properties.Instruction_set
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700466 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
467 if err != nil {
468 ctx.ModuleErrorf("%s", err)
469 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800470
Colin Crossaf19a292015-03-18 12:07:10 -0700471 // TODO: debug
Colin Crossfa138792015-04-24 17:31:52 -0700472 flags.CFlags = append(flags.CFlags, c.Properties.Release.Cflags...)
Colin Crossaf19a292015-03-18 12:07:10 -0700473
Colin Cross28d76592015-03-26 16:14:04 -0700474 if ctx.Host() && !ctx.ContainsProperty("clang") {
Colin Cross97ba0732015-03-23 17:50:24 -0700475 flags.Clang = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800476 }
477
Colin Cross97ba0732015-03-23 17:50:24 -0700478 if flags.Clang {
479 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
Colin Crossfa138792015-04-24 17:31:52 -0700480 flags.CFlags = append(flags.CFlags, c.Properties.Clang_cflags...)
481 flags.AsFlags = append(flags.AsFlags, c.Properties.Clang_asflags...)
Colin Cross97ba0732015-03-23 17:50:24 -0700482 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
483 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
484 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800485
Colin Cross97ba0732015-03-23 17:50:24 -0700486 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
487 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700488 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700489 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700490 }
491
Colin Cross3f40fa42015-01-30 17:27:36 -0800492 target := "-target " + toolchain.ClangTriple()
493 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
494
Colin Cross97ba0732015-03-23 17:50:24 -0700495 flags.CFlags = append(flags.CFlags, target, gccPrefix)
496 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
497 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800498 }
499
Colin Crossfa138792015-04-24 17:31:52 -0700500 if !c.Properties.No_default_compiler_flags {
501 if ctx.Device() && !c.Properties.Allow_undefined_symbols {
Colin Cross97ba0732015-03-23 17:50:24 -0700502 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800503 }
504
Colin Cross56b4d452015-04-21 17:38:44 -0700505 flags.GlobalFlags = append(flags.GlobalFlags, instructionSetFlags)
506
Colin Cross97ba0732015-03-23 17:50:24 -0700507 if flags.Clang {
508 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700509 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800510 toolchain.ClangCflags(),
511 "${commonClangGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700512 fmt.Sprintf("${%sClangGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800513 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700514 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700515 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800516 toolchain.Cflags(),
517 "${commonGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700518 fmt.Sprintf("${%sGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800519 }
520
Colin Crossf6566ed2015-03-24 11:13:38 -0700521 if ctx.Device() {
Colin Crossfa138792015-04-24 17:31:52 -0700522 if c.Properties.Rtti {
Colin Cross97ba0732015-03-23 17:50:24 -0700523 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800524 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700525 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800526 }
527 }
528
Colin Cross97ba0732015-03-23 17:50:24 -0700529 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800530
Colin Cross97ba0732015-03-23 17:50:24 -0700531 if flags.Clang {
532 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
533 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800534 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700535 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
536 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800537 }
Colin Cross28344522015-04-22 13:07:53 -0700538
539 if ctx.Host() {
Colin Crossfa138792015-04-24 17:31:52 -0700540 flags.LdFlags = append(flags.LdFlags, c.Properties.Host_ldlibs...)
Colin Cross28344522015-04-22 13:07:53 -0700541 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800542 }
543
Colin Cross0676e2d2015-04-24 17:39:18 -0700544 flags = c.ccModuleType().flags(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800545
546 // Optimization to reduce size of build.ninja
547 // Replace the long list of flags for each file with a module-local variable
Colin Cross97ba0732015-03-23 17:50:24 -0700548 ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " "))
549 ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " "))
550 ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " "))
551 flags.CFlags = []string{"$cflags"}
552 flags.CppFlags = []string{"$cppflags"}
553 flags.AsFlags = []string{"$asflags"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800554
555 return flags
556}
557
Colin Cross0676e2d2015-04-24 17:39:18 -0700558func (c *CCBase) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross3f40fa42015-01-30 17:27:36 -0800559 return flags
560}
561
562// Compile a list of source files into objects a specified subdirectory
Colin Crossfa138792015-04-24 17:31:52 -0700563func (c *CCBase) customCompileObjs(ctx common.AndroidModuleContext, flags CCFlags,
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700564 subdir string, srcFiles, excludes []string) []string {
Colin Cross581c1892015-04-07 16:50:10 -0700565
566 buildFlags := ccFlagsToBuilderFlags(flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800567
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700568 srcFiles = ctx.ExpandSources(srcFiles, excludes)
Colin Cross581c1892015-04-07 16:50:10 -0700569 srcFiles, deps := genSources(ctx, srcFiles, buildFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800570
Colin Cross581c1892015-04-07 16:50:10 -0700571 return TransformSourceToObj(ctx, subdir, srcFiles, buildFlags, deps)
Colin Cross3f40fa42015-01-30 17:27:36 -0800572}
573
Colin Crossfa138792015-04-24 17:31:52 -0700574// Compile files listed in c.Properties.Srcs into objects
575func (c *CCBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800576
Colin Crossfa138792015-04-24 17:31:52 -0700577 if c.Properties.SkipCompileObjs {
Colin Cross3f40fa42015-01-30 17:27:36 -0800578 return nil
579 }
580
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700581 return c.customCompileObjs(ctx, flags, "", c.Properties.Srcs, c.Properties.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800582}
583
Colin Cross5049f022015-03-18 13:28:46 -0700584// Compile generated source files from dependencies
Colin Crossfa138792015-04-24 17:31:52 -0700585func (c *CCBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700586 var srcs []string
587
Colin Crossfa138792015-04-24 17:31:52 -0700588 if c.Properties.SkipCompileObjs {
Colin Cross5049f022015-03-18 13:28:46 -0700589 return nil
590 }
591
592 ctx.VisitDirectDeps(func(module blueprint.Module) {
593 if gen, ok := module.(genrule.SourceFileGenerator); ok {
594 srcs = append(srcs, gen.GeneratedSourceFiles()...)
595 }
596 })
597
598 if len(srcs) == 0 {
599 return nil
600 }
601
Colin Cross581c1892015-04-07 16:50:10 -0700602 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags), nil)
Colin Cross5049f022015-03-18 13:28:46 -0700603}
604
Colin Crossfa138792015-04-24 17:31:52 -0700605func (c *CCBase) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800606 return ""
607}
608
Colin Crossfa138792015-04-24 17:31:52 -0700609func (c *CCBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800610 names []string) (modules []common.AndroidModule,
Colin Cross28344522015-04-22 13:07:53 -0700611 outputFiles []string, exportedFlags []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800612
613 for _, n := range names {
614 found := false
615 ctx.VisitDirectDeps(func(m blueprint.Module) {
616 otherName := ctx.OtherModuleName(m)
617 if otherName != n {
618 return
619 }
620
Colin Cross97ba0732015-03-23 17:50:24 -0700621 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800622 if a.Disabled() {
623 // If a cc_library host+device module depends on a library that exists as both
624 // cc_library_shared and cc_library_host_shared, it will end up with two
625 // dependencies with the same name, one of which is marked disabled for each
626 // of host and device. Ignore the disabled one.
627 return
628 }
Colin Crossd3ba0392015-05-07 14:11:29 -0700629 if a.HostOrDevice() != ctx.HostOrDevice() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800630 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
631 otherName)
632 return
633 }
634
635 if outputFile := a.outputFile(); outputFile != "" {
636 if found {
637 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
638 return
639 }
640 outputFiles = append(outputFiles, outputFile)
641 modules = append(modules, a)
Colin Cross28344522015-04-22 13:07:53 -0700642 if i, ok := a.(ccExportedFlagsProducer); ok {
643 exportedFlags = append(exportedFlags, i.exportedFlags()...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800644 }
645 found = true
646 } else {
647 ctx.ModuleErrorf("module %q missing output file", otherName)
648 return
649 }
650 } else {
651 ctx.ModuleErrorf("module %q not an android module", otherName)
652 return
653 }
654 })
655 if !found {
656 ctx.ModuleErrorf("unsatisified dependency on %q", n)
657 }
658 }
659
Colin Cross28344522015-04-22 13:07:53 -0700660 return modules, outputFiles, exportedFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800661}
662
Colin Cross21b9a242015-03-24 14:15:58 -0700663// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
664// containing paths
Colin Crossfa138792015-04-24 17:31:52 -0700665func (c *CCBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700666 var depPaths CCDeps
Colin Cross28344522015-04-22 13:07:53 -0700667 var newCflags []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800668
Colin Cross21b9a242015-03-24 14:15:58 -0700669 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800670
Colin Cross28344522015-04-22 13:07:53 -0700671 wholeStaticLibModules, depPaths.WholeStaticLibs, newCflags =
Colin Cross21b9a242015-03-24 14:15:58 -0700672 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
Colin Cross28344522015-04-22 13:07:53 -0700673 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800674
Colin Cross21b9a242015-03-24 14:15:58 -0700675 for _, m := range wholeStaticLibModules {
676 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
677 depPaths.WholeStaticLibObjFiles =
678 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
679 } else {
680 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
681 }
682 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800683
Colin Cross28344522015-04-22 13:07:53 -0700684 _, depPaths.StaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.StaticLibs)
685 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700686
Colin Cross28344522015-04-22 13:07:53 -0700687 _, depPaths.LateStaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
688 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700689
Colin Cross28344522015-04-22 13:07:53 -0700690 _, depPaths.SharedLibs, newCflags = c.depsToPathsFromList(ctx, depNames.SharedLibs)
691 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700692
693 ctx.VisitDirectDeps(func(m blueprint.Module) {
Dan Albertc3144b12015-04-28 18:17:56 -0700694 if obj, ok := m.(ccObjectProvider); ok {
Colin Cross21b9a242015-03-24 14:15:58 -0700695 otherName := ctx.OtherModuleName(m)
696 if otherName == depNames.CrtBegin {
Colin Crossfa138792015-04-24 17:31:52 -0700697 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700698 depPaths.CrtBegin = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700699 }
700 } else if otherName == depNames.CrtEnd {
Colin Crossfa138792015-04-24 17:31:52 -0700701 if !c.Properties.Nocrt {
Dan Albertc3144b12015-04-28 18:17:56 -0700702 depPaths.CrtEnd = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700703 }
704 } else {
Dan Albertc3144b12015-04-28 18:17:56 -0700705 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.object().outputFile())
Colin Cross21b9a242015-03-24 14:15:58 -0700706 }
707 }
708 })
709
710 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800711}
712
Colin Cross7d5136f2015-05-11 13:39:40 -0700713type ccLinkedProperties struct {
714 VariantIsShared bool `blueprint:"mutated"`
715 VariantIsStatic bool `blueprint:"mutated"`
716 VariantIsStaticBinary bool `blueprint:"mutated"`
717}
718
Colin Crossfa138792015-04-24 17:31:52 -0700719// CCLinked contains the properties and members used by libraries and executables
720type CCLinked struct {
721 CCBase
Colin Cross7d5136f2015-05-11 13:39:40 -0700722 dynamicProperties ccLinkedProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800723}
724
Colin Crossfa138792015-04-24 17:31:52 -0700725func newCCDynamic(dynamic *CCLinked, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700726 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
727
Colin Crossed4cf0b2015-03-26 14:43:45 -0700728 props = append(props, &dynamic.dynamicProperties)
729
Colin Crossfa138792015-04-24 17:31:52 -0700730 return newCCBase(&dynamic.CCBase, module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700731}
732
Colin Crossfa138792015-04-24 17:31:52 -0700733func (c *CCLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross28d76592015-03-26 16:14:04 -0700734 if ctx.ContainsProperty("system_shared_libs") {
Colin Crossfa138792015-04-24 17:31:52 -0700735 return c.Properties.System_shared_libs
736 } else if ctx.Device() && c.Properties.Sdk_version == "" {
Colin Cross577f6e42015-03-27 18:23:34 -0700737 return []string{"libc", "libm"}
Colin Cross28d76592015-03-26 16:14:04 -0700738 } else {
Colin Cross577f6e42015-03-27 18:23:34 -0700739 return nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800740 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800741}
742
Colin Crossfa138792015-04-24 17:31:52 -0700743func (c *CCLinked) stl(ctx common.AndroidBaseContext) string {
744 if c.Properties.Sdk_version != "" && ctx.Device() {
745 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700746 case "":
747 return "ndk_system"
748 case "c++_shared", "c++_static",
749 "stlport_shared", "stlport_static",
750 "gnustl_static":
Colin Crossfa138792015-04-24 17:31:52 -0700751 return "ndk_lib" + c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700752 default:
Colin Crossfa138792015-04-24 17:31:52 -0700753 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700754 return ""
755 }
756 }
757
Colin Crossfa138792015-04-24 17:31:52 -0700758 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700759 case "libc++", "libc++_static",
760 "stlport", "stlport_static",
761 "libstdc++":
Colin Crossfa138792015-04-24 17:31:52 -0700762 return c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700763 case "none":
764 return ""
765 case "":
Colin Cross18b6dc52015-04-28 13:20:37 -0700766 if c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700767 return "libc++_static"
Colin Cross18b6dc52015-04-28 13:20:37 -0700768 } else {
769 return "libc++" // TODO: mingw needs libstdc++
Colin Crossed4cf0b2015-03-26 14:43:45 -0700770 }
771 default:
Colin Crossfa138792015-04-24 17:31:52 -0700772 ctx.ModuleErrorf("stl: %q is not a supported STL", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700773 return ""
774 }
775}
776
Colin Cross0af4b842015-04-30 16:36:18 -0700777var hostDynamicGccLibs, hostStaticGccLibs []string
778
779func init() {
780 if runtime.GOOS == "darwin" {
781 hostDynamicGccLibs = []string{"-lc", "-lSystem"}
782 hostStaticGccLibs = []string{"NO_STATIC_HOST_BINARIES_ON_DARWIN"}
783 } else {
784 hostDynamicGccLibs = []string{"-lgcc_s", "-lgcc", "-lc", "-lgcc_s", "-lgcc"}
785 hostStaticGccLibs = []string{"-Wl,--start-group", "-lgcc", "-lgcc_eh", "-lc", "-Wl,--end-group"}
786 }
787}
Colin Cross712fc022015-04-27 11:13:34 -0700788
Colin Crosse11befc2015-04-27 17:49:17 -0700789func (c *CCLinked) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700790 stl := c.stl(ctx)
791 if ctx.Failed() {
792 return flags
793 }
794
795 switch stl {
796 case "libc++", "libc++_static":
797 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
Colin Cross28344522015-04-22 13:07:53 -0700798 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/external/libcxx/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700799 if ctx.Host() {
800 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
801 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross712fc022015-04-27 11:13:34 -0700802 flags.LdFlags = append(flags.LdFlags, "-lm", "-lpthread")
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 case "stlport", "stlport_static":
810 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700811 flags.CFlags = append(flags.CFlags,
812 "-I${SrcDir}/external/stlport/stlport",
813 "-I${SrcDir}/bionic/libstdc++/include",
814 "-I${SrcDir}/bionic")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700815 }
816 case "libstdc++":
817 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
818 // tree is in good enough shape to not need it.
819 // Host builds will use GNU libstdc++.
820 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700821 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/bionic/libstdc++/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700822 }
823 case "ndk_system":
Colin Cross1332b002015-04-07 17:11:30 -0700824 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources/"
Colin Cross28344522015-04-22 13:07:53 -0700825 flags.CFlags = append(flags.CFlags, "-isystem "+ndkSrcRoot+"cxx-stl/system/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700826 case "ndk_libc++_shared", "ndk_libc++_static":
827 // TODO(danalbert): This really shouldn't be here...
828 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
829 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
830 // Nothing
831 case "":
832 // None or error.
833 if ctx.Host() {
834 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
835 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross18b6dc52015-04-28 13:20:37 -0700836 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700837 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700838 } else {
839 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700840 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700841 }
842 default:
Colin Crossfa138792015-04-24 17:31:52 -0700843 panic(fmt.Errorf("Unknown stl in CCLinked.Flags: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700844 }
845
846 return flags
847}
848
Colin Crosse11befc2015-04-27 17:49:17 -0700849func (c *CCLinked) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
850 depNames = c.CCBase.depNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800851
Colin Crossed4cf0b2015-03-26 14:43:45 -0700852 stl := c.stl(ctx)
853 if ctx.Failed() {
854 return depNames
855 }
856
857 switch stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700858 case "libstdc++":
859 if ctx.Device() {
860 depNames.SharedLibs = append(depNames.SharedLibs, stl)
861 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700862 case "libc++", "libc++_static":
863 if stl == "libc++" {
864 depNames.SharedLibs = append(depNames.SharedLibs, stl)
865 } else {
866 depNames.StaticLibs = append(depNames.StaticLibs, stl)
867 }
868 if ctx.Device() {
869 if ctx.Arch().ArchType == common.Arm {
870 depNames.StaticLibs = append(depNames.StaticLibs, "libunwind_llvm")
871 }
872 if c.staticBinary() {
873 depNames.StaticLibs = append(depNames.StaticLibs, "libdl")
874 } else {
875 depNames.SharedLibs = append(depNames.SharedLibs, "libdl")
876 }
877 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700878 case "stlport":
879 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
880 case "stlport_static":
881 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
882 case "":
883 // None or error.
884 case "ndk_system":
885 // TODO: Make a system STL prebuilt for the NDK.
886 // 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 -0700887 // its own includes. The includes are handled in CCBase.Flags().
Colin Cross577f6e42015-03-27 18:23:34 -0700888 depNames.SharedLibs = append([]string{"libstdc++"}, depNames.SharedLibs...)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700889 case "ndk_libc++_shared", "ndk_libstlport_shared":
890 depNames.SharedLibs = append(depNames.SharedLibs, stl)
891 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
892 depNames.StaticLibs = append(depNames.StaticLibs, stl)
893 default:
Colin Crosse11befc2015-04-27 17:49:17 -0700894 panic(fmt.Errorf("Unknown stl in CCLinked.depNames: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700895 }
896
Colin Cross74d1ec02015-04-28 13:30:13 -0700897 if ctx.ModuleName() != "libcompiler_rt-extras" {
898 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
899 }
900
Colin Crossf6566ed2015-03-24 11:13:38 -0700901 if ctx.Device() {
Colin Cross77b00fa2015-03-16 16:15:49 -0700902 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700903 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700904
Colin Cross18b6dc52015-04-28 13:20:37 -0700905 if !c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700906 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
907 }
Colin Cross577f6e42015-03-27 18:23:34 -0700908
Colin Crossfa138792015-04-24 17:31:52 -0700909 if c.Properties.Sdk_version != "" {
910 version := c.Properties.Sdk_version
Colin Cross577f6e42015-03-27 18:23:34 -0700911 depNames.SharedLibs = append(depNames.SharedLibs,
912 "ndk_libc."+version,
913 "ndk_libm."+version,
914 )
915 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800916 }
917
Colin Cross21b9a242015-03-24 14:15:58 -0700918 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800919}
920
Colin Crossed4cf0b2015-03-26 14:43:45 -0700921// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
922type ccLinkedInterface interface {
923 // Returns true if the build options for the module have selected a static or shared build
924 buildStatic() bool
925 buildShared() bool
926
927 // Sets whether a specific variant is static or shared
Colin Cross18b6dc52015-04-28 13:20:37 -0700928 setStatic(bool)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700929
Colin Cross18b6dc52015-04-28 13:20:37 -0700930 // Returns whether a specific variant is a static library or binary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700931 static() bool
Colin Cross18b6dc52015-04-28 13:20:37 -0700932
933 // Returns whether a module is a static binary
934 staticBinary() bool
Colin Crossed4cf0b2015-03-26 14:43:45 -0700935}
936
937var _ ccLinkedInterface = (*CCLibrary)(nil)
938var _ ccLinkedInterface = (*CCBinary)(nil)
939
Colin Crossfa138792015-04-24 17:31:52 -0700940func (c *CCLinked) static() bool {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700941 return c.dynamicProperties.VariantIsStatic
942}
943
Colin Cross18b6dc52015-04-28 13:20:37 -0700944func (c *CCLinked) staticBinary() bool {
945 return c.dynamicProperties.VariantIsStaticBinary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700946}
947
Colin Cross18b6dc52015-04-28 13:20:37 -0700948func (c *CCLinked) setStatic(static bool) {
949 c.dynamicProperties.VariantIsStatic = static
Colin Crossed4cf0b2015-03-26 14:43:45 -0700950}
951
Colin Cross28344522015-04-22 13:07:53 -0700952type ccExportedFlagsProducer interface {
953 exportedFlags() []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800954}
955
956//
957// Combined static+shared libraries
958//
959
Colin Cross7d5136f2015-05-11 13:39:40 -0700960type CCLibraryProperties struct {
961 BuildStatic bool `blueprint:"mutated"`
962 BuildShared bool `blueprint:"mutated"`
963 Static struct {
964 Srcs []string `android:"arch_variant"`
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700965 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700966 Cflags []string `android:"arch_variant"`
967 Whole_static_libs []string `android:"arch_variant"`
968 Static_libs []string `android:"arch_variant"`
969 Shared_libs []string `android:"arch_variant"`
970 } `android:"arch_variant"`
971 Shared struct {
972 Srcs []string `android:"arch_variant"`
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700973 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700974 Cflags []string `android:"arch_variant"`
975 Whole_static_libs []string `android:"arch_variant"`
976 Static_libs []string `android:"arch_variant"`
977 Shared_libs []string `android:"arch_variant"`
978 } `android:"arch_variant"`
Colin Crossaee540a2015-07-06 17:48:31 -0700979
980 // local file name to pass to the linker as --version_script
981 Version_script string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700982}
983
Colin Cross97ba0732015-03-23 17:50:24 -0700984type CCLibrary struct {
Colin Crossfa138792015-04-24 17:31:52 -0700985 CCLinked
Colin Cross3f40fa42015-01-30 17:27:36 -0800986
Colin Cross28344522015-04-22 13:07:53 -0700987 reuseFrom ccLibraryInterface
988 reuseObjFiles []string
989 objFiles []string
990 exportFlags []string
991 out string
Colin Cross3f40fa42015-01-30 17:27:36 -0800992
Colin Cross7d5136f2015-05-11 13:39:40 -0700993 LibraryProperties CCLibraryProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800994}
995
Colin Crossed4cf0b2015-03-26 14:43:45 -0700996func (c *CCLibrary) buildStatic() bool {
997 return c.LibraryProperties.BuildStatic
998}
999
1000func (c *CCLibrary) buildShared() bool {
1001 return c.LibraryProperties.BuildShared
1002}
1003
Colin Cross97ba0732015-03-23 17:50:24 -07001004type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001005 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -07001006 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -07001007 setReuseFrom(ccLibraryInterface)
1008 getReuseFrom() ccLibraryInterface
1009 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -07001010 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -07001011}
1012
Colin Crossed4cf0b2015-03-26 14:43:45 -07001013var _ ccLibraryInterface = (*CCLibrary)(nil)
1014
Colin Cross97ba0732015-03-23 17:50:24 -07001015func (c *CCLibrary) ccLibrary() *CCLibrary {
1016 return c
Colin Cross3f40fa42015-01-30 17:27:36 -08001017}
1018
Colin Cross97ba0732015-03-23 17:50:24 -07001019func NewCCLibrary(library *CCLibrary, module CCModuleType,
1020 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
1021
Colin Crossfa138792015-04-24 17:31:52 -07001022 return newCCDynamic(&library.CCLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -07001023 &library.LibraryProperties)
1024}
1025
1026func CCLibraryFactory() (blueprint.Module, []interface{}) {
1027 module := &CCLibrary{}
1028
1029 module.LibraryProperties.BuildShared = true
1030 module.LibraryProperties.BuildStatic = true
1031
1032 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
1033}
1034
Colin Cross0676e2d2015-04-24 17:39:18 -07001035func (c *CCLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001036 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Cross2732e9a2015-04-28 13:23:52 -07001037 if c.static() {
1038 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Static.Whole_static_libs...)
1039 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Static.Static_libs...)
1040 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Static.Shared_libs...)
1041 } else {
Colin Crossf6566ed2015-03-24 11:13:38 -07001042 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001043 if c.Properties.Sdk_version == "" {
1044 depNames.CrtBegin = "crtbegin_so"
1045 depNames.CrtEnd = "crtend_so"
1046 } else {
1047 depNames.CrtBegin = "ndk_crtbegin_so." + c.Properties.Sdk_version
1048 depNames.CrtEnd = "ndk_crtend_so." + c.Properties.Sdk_version
1049 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001050 }
Colin Cross2732e9a2015-04-28 13:23:52 -07001051 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Shared.Whole_static_libs...)
1052 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Shared.Static_libs...)
1053 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Shared.Shared_libs...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001054 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001055
Colin Cross21b9a242015-03-24 14:15:58 -07001056 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001057}
1058
Colin Cross97ba0732015-03-23 17:50:24 -07001059func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001060 return c.out
1061}
1062
Colin Crossed4cf0b2015-03-26 14:43:45 -07001063func (c *CCLibrary) getReuseObjFiles() []string {
1064 return c.reuseObjFiles
1065}
1066
1067func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
1068 c.reuseFrom = reuseFrom
1069}
1070
1071func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
1072 return c.reuseFrom
1073}
1074
Colin Cross97ba0732015-03-23 17:50:24 -07001075func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001076 return c.objFiles
1077}
1078
Colin Cross28344522015-04-22 13:07:53 -07001079func (c *CCLibrary) exportedFlags() []string {
1080 return c.exportFlags
Colin Cross3f40fa42015-01-30 17:27:36 -08001081}
1082
Colin Cross0676e2d2015-04-24 17:39:18 -07001083func (c *CCLibrary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001084 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001085
Colin Cross97ba0732015-03-23 17:50:24 -07001086 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -08001087
Colin Crossd8e780d2015-04-28 17:39:43 -07001088 if c.static() {
1089 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Static.Cflags...)
1090 } else {
1091 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Shared.Cflags...)
1092 }
1093
Colin Cross18b6dc52015-04-28 13:20:37 -07001094 if !c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001095 libName := ctx.ModuleName()
1096 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
1097 sharedFlag := "-Wl,-shared"
Colin Crossfa138792015-04-24 17:31:52 -07001098 if c.Properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001099 sharedFlag = "-shared"
1100 }
Colin Crossf6566ed2015-03-24 11:13:38 -07001101 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001102 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001103 }
Colin Cross97ba0732015-03-23 17:50:24 -07001104
Colin Cross0af4b842015-04-30 16:36:18 -07001105 if ctx.Darwin() {
1106 flags.LdFlags = append(flags.LdFlags,
1107 "-dynamiclib",
1108 "-single_module",
1109 //"-read_only_relocs suppress",
1110 "-install_name @rpath/"+libName+sharedLibraryExtension,
1111 )
1112 } else {
1113 flags.LdFlags = append(flags.LdFlags,
1114 "-Wl,--gc-sections",
1115 sharedFlag,
1116 "-Wl,-soname,"+libName+sharedLibraryExtension,
1117 )
1118 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001119 }
Colin Cross97ba0732015-03-23 17:50:24 -07001120
1121 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001122}
1123
Colin Cross97ba0732015-03-23 17:50:24 -07001124func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1125 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001126
1127 staticFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001128 objFilesStatic := c.customCompileObjs(ctx, staticFlags, common.DeviceStaticLibrary,
Dan Willemsen2ef08f42015-06-30 18:15:24 -07001129 c.LibraryProperties.Static.Srcs, c.LibraryProperties.Static.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001130
1131 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001132 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001133
1134 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1135
Colin Cross0af4b842015-04-30 16:36:18 -07001136 if ctx.Darwin() {
1137 TransformDarwinObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1138 } else {
1139 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1140 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001141
1142 c.objFiles = objFiles
1143 c.out = outputFile
Colin Crossf2298272015-05-12 11:36:53 -07001144
1145 common.CheckModuleSrcDirsExist(ctx, c.Properties.Export_include_dirs, "export_include_dirs")
Colin Crossfa138792015-04-24 17:31:52 -07001146 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001147 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001148
1149 ctx.CheckbuildFile(outputFile)
1150}
1151
Colin Cross97ba0732015-03-23 17:50:24 -07001152func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1153 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001154
1155 sharedFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001156 objFilesShared := c.customCompileObjs(ctx, sharedFlags, common.DeviceSharedLibrary,
Dan Willemsen2ef08f42015-06-30 18:15:24 -07001157 c.LibraryProperties.Shared.Srcs, c.LibraryProperties.Shared.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001158
1159 objFiles = append(objFiles, objFilesShared...)
1160
1161 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1162
Colin Crossaee540a2015-07-06 17:48:31 -07001163 var linkerDeps []string
1164
1165 if c.LibraryProperties.Version_script != "" {
1166 versionScript := filepath.Join(common.ModuleSrcDir(ctx), c.LibraryProperties.Version_script)
1167 sharedFlags.LdFlags = append(sharedFlags.LdFlags, "-Wl,--version-script,"+versionScript)
1168 linkerDeps = append(linkerDeps, versionScript)
1169 }
1170
Colin Cross97ba0732015-03-23 17:50:24 -07001171 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossaee540a2015-07-06 17:48:31 -07001172 deps.LateStaticLibs, deps.WholeStaticLibs, linkerDeps, deps.CrtBegin, deps.CrtEnd, false,
Colin Cross77b00fa2015-03-16 16:15:49 -07001173 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001174
1175 c.out = outputFile
Colin Crossfa138792015-04-24 17:31:52 -07001176 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001177 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Cross3f40fa42015-01-30 17:27:36 -08001178}
1179
Colin Cross97ba0732015-03-23 17:50:24 -07001180func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1181 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001182
1183 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001184 if c.getReuseFrom().ccLibrary() == c {
1185 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001186 } else {
Colin Cross2732e9a2015-04-28 13:23:52 -07001187 if c.getReuseFrom().ccLibrary().LibraryProperties.Static.Cflags == nil &&
1188 c.LibraryProperties.Shared.Cflags == nil {
1189 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
1190 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001191 }
1192
Colin Crossed4cf0b2015-03-26 14:43:45 -07001193 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001194 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1195 } else {
1196 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1197 }
1198}
1199
Colin Cross97ba0732015-03-23 17:50:24 -07001200func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001201 // Static libraries do not get installed.
1202}
1203
Colin Cross97ba0732015-03-23 17:50:24 -07001204func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001205 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001206 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001207 installDir = "lib64"
1208 }
1209
Colin Crossfa138792015-04-24 17:31:52 -07001210 ctx.InstallFile(filepath.Join(installDir, c.Properties.Relative_install_path), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001211}
1212
Colin Cross97ba0732015-03-23 17:50:24 -07001213func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001214 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001215 c.installStaticLibrary(ctx, flags)
1216 } else {
1217 c.installSharedLibrary(ctx, flags)
1218 }
1219}
1220
Colin Cross3f40fa42015-01-30 17:27:36 -08001221//
1222// Objects (for crt*.o)
1223//
1224
Dan Albertc3144b12015-04-28 18:17:56 -07001225type ccObjectProvider interface {
1226 object() *ccObject
1227}
1228
Colin Cross3f40fa42015-01-30 17:27:36 -08001229type ccObject struct {
Colin Crossfa138792015-04-24 17:31:52 -07001230 CCBase
Colin Cross3f40fa42015-01-30 17:27:36 -08001231 out string
1232}
1233
Dan Albertc3144b12015-04-28 18:17:56 -07001234func (c *ccObject) object() *ccObject {
1235 return c
1236}
1237
Colin Cross97ba0732015-03-23 17:50:24 -07001238func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001239 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001240
Colin Crossfa138792015-04-24 17:31:52 -07001241 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001242}
1243
1244func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1245 // object files can't have any dynamic dependencies
1246 return nil
1247}
1248
Colin Cross0676e2d2015-04-24 17:39:18 -07001249func (*ccObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001250 // object files can't have any dynamic dependencies
1251 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001252}
1253
1254func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001255 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001256
Colin Cross97ba0732015-03-23 17:50:24 -07001257 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001258
1259 var outputFile string
1260 if len(objFiles) == 1 {
1261 outputFile = objFiles[0]
1262 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001263 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+objectExtension)
Colin Cross3f40fa42015-01-30 17:27:36 -08001264 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1265 }
1266
1267 c.out = outputFile
1268
1269 ctx.CheckbuildFile(outputFile)
1270}
1271
Colin Cross97ba0732015-03-23 17:50:24 -07001272func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001273 // Object files do not get installed.
1274}
1275
Colin Cross3f40fa42015-01-30 17:27:36 -08001276func (c *ccObject) outputFile() string {
1277 return c.out
1278}
1279
Dan Albertc3144b12015-04-28 18:17:56 -07001280var _ ccObjectProvider = (*ccObject)(nil)
1281
Colin Cross3f40fa42015-01-30 17:27:36 -08001282//
1283// Executables
1284//
1285
Colin Cross7d5136f2015-05-11 13:39:40 -07001286type CCBinaryProperties struct {
1287 // compile executable with -static
1288 Static_executable bool
1289
1290 // set the name of the output
1291 Stem string `android:"arch_variant"`
1292
1293 // append to the name of the output
1294 Suffix string `android:"arch_variant"`
1295
1296 // if set, add an extra objcopy --prefix-symbols= step
1297 Prefix_symbols string
1298}
1299
Colin Cross97ba0732015-03-23 17:50:24 -07001300type CCBinary struct {
Colin Crossfa138792015-04-24 17:31:52 -07001301 CCLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001302 out string
Colin Crossd350ecd2015-04-28 13:25:36 -07001303 installFile string
Colin Cross7d5136f2015-05-11 13:39:40 -07001304 BinaryProperties CCBinaryProperties
Colin Cross3f40fa42015-01-30 17:27:36 -08001305}
1306
Colin Crossed4cf0b2015-03-26 14:43:45 -07001307func (c *CCBinary) buildStatic() bool {
1308 return c.BinaryProperties.Static_executable
1309}
1310
1311func (c *CCBinary) buildShared() bool {
1312 return !c.BinaryProperties.Static_executable
1313}
1314
Colin Cross97ba0732015-03-23 17:50:24 -07001315func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001316 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001317 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001318 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001319 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001320
1321 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001322}
1323
Colin Cross0676e2d2015-04-24 17:39:18 -07001324func (c *CCBinary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001325 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001326 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001327 if c.Properties.Sdk_version == "" {
1328 if c.BinaryProperties.Static_executable {
1329 depNames.CrtBegin = "crtbegin_static"
1330 } else {
1331 depNames.CrtBegin = "crtbegin_dynamic"
1332 }
1333 depNames.CrtEnd = "crtend_android"
Colin Cross3f40fa42015-01-30 17:27:36 -08001334 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001335 if c.BinaryProperties.Static_executable {
1336 depNames.CrtBegin = "ndk_crtbegin_static." + c.Properties.Sdk_version
1337 } else {
1338 depNames.CrtBegin = "ndk_crtbegin_dynamic." + c.Properties.Sdk_version
1339 }
1340 depNames.CrtEnd = "ndk_crtend_android." + c.Properties.Sdk_version
Colin Cross3f40fa42015-01-30 17:27:36 -08001341 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001342
1343 if c.BinaryProperties.Static_executable {
Colin Cross74d1ec02015-04-28 13:30:13 -07001344 if c.stl(ctx) == "libc++_static" {
1345 depNames.StaticLibs = append(depNames.StaticLibs, "libm", "libc", "libdl")
1346 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001347 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1348 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1349 // move them to the beginning of deps.LateStaticLibs
1350 var groupLibs []string
1351 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1352 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1353 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1354 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001355 }
Colin Cross21b9a242015-03-24 14:15:58 -07001356 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001357}
1358
Colin Cross97ba0732015-03-23 17:50:24 -07001359func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001360 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001361
Colin Cross1f8f2342015-03-26 16:09:47 -07001362 props = append(props, &binary.BinaryProperties)
1363
Colin Crossfa138792015-04-24 17:31:52 -07001364 return newCCDynamic(&binary.CCLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001365}
1366
Colin Cross97ba0732015-03-23 17:50:24 -07001367func CCBinaryFactory() (blueprint.Module, []interface{}) {
1368 module := &CCBinary{}
1369
1370 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001371}
1372
Colin Cross18b6dc52015-04-28 13:20:37 -07001373func (c *CCBinary) ModifyProperties(ctx common.AndroidBaseContext) {
Colin Cross0af4b842015-04-30 16:36:18 -07001374 if ctx.Darwin() {
1375 c.BinaryProperties.Static_executable = false
1376 }
Colin Cross18b6dc52015-04-28 13:20:37 -07001377 if c.BinaryProperties.Static_executable {
1378 c.dynamicProperties.VariantIsStaticBinary = true
1379 }
1380}
1381
Colin Cross0676e2d2015-04-24 17:39:18 -07001382func (c *CCBinary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001383 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001384
Colin Cross97ba0732015-03-23 17:50:24 -07001385 flags.CFlags = append(flags.CFlags, "-fpie")
1386
Colin Crossf6566ed2015-03-24 11:13:38 -07001387 if ctx.Device() {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001388 if c.BinaryProperties.Static_executable {
1389 // Clang driver needs -static to create static executable.
1390 // However, bionic/linker uses -shared to overwrite.
1391 // Linker for x86 targets does not allow coexistance of -static and -shared,
1392 // so we add -static only if -shared is not used.
1393 if !inList("-shared", flags.LdFlags) {
1394 flags.LdFlags = append(flags.LdFlags, "-static")
1395 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001396
Colin Crossed4cf0b2015-03-26 14:43:45 -07001397 flags.LdFlags = append(flags.LdFlags,
1398 "-nostdlib",
1399 "-Bstatic",
1400 "-Wl,--gc-sections",
1401 )
1402
1403 } else {
1404 linker := "/system/bin/linker"
1405 if flags.Toolchain.Is64Bit() {
1406 linker = "/system/bin/linker64"
1407 }
1408
1409 flags.LdFlags = append(flags.LdFlags,
1410 "-nostdlib",
1411 "-Bdynamic",
1412 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1413 "-Wl,--gc-sections",
1414 "-Wl,-z,nocopyreloc",
1415 )
1416 }
Colin Cross0af4b842015-04-30 16:36:18 -07001417 } else if ctx.Darwin() {
1418 flags.LdFlags = append(flags.LdFlags, "-Wl,-headerpad_max_install_names")
Colin Cross3f40fa42015-01-30 17:27:36 -08001419 }
1420
Colin Cross97ba0732015-03-23 17:50:24 -07001421 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001422}
1423
Colin Cross97ba0732015-03-23 17:50:24 -07001424func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1425 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001426
Colin Crossfa138792015-04-24 17:31:52 -07001427 if !c.BinaryProperties.Static_executable && inList("libc", c.Properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001428 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1429 "from static libs or set static_executable: true")
1430 }
1431
1432 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001433 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001434 if c.BinaryProperties.Prefix_symbols != "" {
1435 afterPrefixSymbols := outputFile
1436 outputFile = outputFile + ".intermediate"
1437 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1438 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1439 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001440
Colin Crossaee540a2015-07-06 17:48:31 -07001441 var linkerDeps []string
1442
Colin Cross97ba0732015-03-23 17:50:24 -07001443 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossaee540a2015-07-06 17:48:31 -07001444 deps.LateStaticLibs, deps.WholeStaticLibs, linkerDeps, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001445 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001446}
Colin Cross3f40fa42015-01-30 17:27:36 -08001447
Colin Cross97ba0732015-03-23 17:50:24 -07001448func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossd350ecd2015-04-28 13:25:36 -07001449 c.installFile = ctx.InstallFile(filepath.Join("bin", c.Properties.Relative_install_path), c.out)
1450}
1451
1452func (c *CCBinary) HostToolPath() string {
1453 if c.HostOrDevice().Host() {
1454 return c.installFile
1455 }
1456 return ""
Dan Albertc403f7c2015-03-18 14:01:18 -07001457}
1458
Colin Cross7d5136f2015-05-11 13:39:40 -07001459type CCTestProperties struct {
1460 // Create a separate test for each source file. Useful when there is
1461 // global state that can not be torn down and reset between each test suite.
1462 Test_per_src bool
1463}
1464
Colin Cross9ffb4f52015-04-24 17:48:09 -07001465type CCTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001466 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001467
Colin Cross7d5136f2015-05-11 13:39:40 -07001468 TestProperties CCTestProperties
Dan Albertc403f7c2015-03-18 14:01:18 -07001469}
1470
Colin Cross9ffb4f52015-04-24 17:48:09 -07001471func (c *CCTest) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross0676e2d2015-04-24 17:39:18 -07001472 flags = c.CCBinary.flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001473
Colin Cross97ba0732015-03-23 17:50:24 -07001474 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001475 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001476 flags.CFlags = append(flags.CFlags, "-O0", "-g")
Colin Cross28344522015-04-22 13:07:53 -07001477 flags.LdFlags = append(flags.LdFlags, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001478 }
1479
1480 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross28344522015-04-22 13:07:53 -07001481 flags.CFlags = append(flags.CFlags,
1482 "-I"+filepath.Join(ctx.AConfig().SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001483
Colin Cross21b9a242015-03-24 14:15:58 -07001484 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001485}
1486
Colin Cross9ffb4f52015-04-24 17:48:09 -07001487func (c *CCTest) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001488 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
Colin Crossa8a93d32015-04-28 13:26:49 -07001489 depNames = c.CCBinary.depNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -07001490 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001491}
1492
Colin Cross9ffb4f52015-04-24 17:48:09 -07001493func (c *CCTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001494 if ctx.Device() {
Colin Crossa8a93d32015-04-28 13:26:49 -07001495 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001496 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001497 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001498 }
1499}
1500
Colin Cross9ffb4f52015-04-24 17:48:09 -07001501func (c *CCTest) testPerSrc() bool {
1502 return c.TestProperties.Test_per_src
Colin Cross6b290692015-03-19 14:05:33 -07001503}
1504
Colin Cross9ffb4f52015-04-24 17:48:09 -07001505func (c *CCTest) test() *CCTest {
1506 return c
1507}
1508
1509func NewCCTest(test *CCTest, module CCModuleType,
1510 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1511
1512 props = append(props, &test.TestProperties)
1513
1514 return NewCCBinary(&test.CCBinary, module, hod, props...)
1515}
1516
1517func CCTestFactory() (blueprint.Module, []interface{}) {
1518 module := &CCTest{}
1519
1520 return NewCCTest(module, module, common.HostAndDeviceSupported)
1521}
1522
1523type testPerSrc interface {
1524 test() *CCTest
1525 testPerSrc() bool
1526}
1527
1528var _ testPerSrc = (*CCTest)(nil)
1529
Colin Cross6b290692015-03-19 14:05:33 -07001530func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001531 if test, ok := mctx.Module().(testPerSrc); ok {
1532 if test.testPerSrc() {
1533 testNames := make([]string, len(test.test().Properties.Srcs))
1534 for i, src := range test.test().Properties.Srcs {
Colin Cross6b290692015-03-19 14:05:33 -07001535 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1536 }
1537 tests := mctx.CreateLocalVariations(testNames...)
Colin Cross9ffb4f52015-04-24 17:48:09 -07001538 for i, src := range test.test().Properties.Srcs {
1539 tests[i].(testPerSrc).test().Properties.Srcs = []string{src}
1540 tests[i].(testPerSrc).test().BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001541 }
1542 }
1543 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001544}
1545
Colin Cross2ba19d92015-05-07 15:44:20 -07001546type CCBenchmark struct {
1547 CCBinary
1548}
1549
1550func (c *CCBenchmark) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1551 depNames = c.CCBinary.depNames(ctx, depNames)
1552 depNames.StaticLibs = append(depNames.StaticLibs, "libbenchmark")
1553 return depNames
1554}
1555
1556func (c *CCBenchmark) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1557 if ctx.Device() {
1558 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
1559 } else {
1560 c.CCBinary.installModule(ctx, flags)
1561 }
1562}
1563
1564func NewCCBenchmark(test *CCBenchmark, module CCModuleType,
1565 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1566
1567 return NewCCBinary(&test.CCBinary, module, hod, props...)
1568}
1569
1570func CCBenchmarkFactory() (blueprint.Module, []interface{}) {
1571 module := &CCBenchmark{}
1572
1573 return NewCCBenchmark(module, module, common.HostAndDeviceSupported)
1574}
1575
Colin Cross3f40fa42015-01-30 17:27:36 -08001576//
1577// Static library
1578//
1579
Colin Cross97ba0732015-03-23 17:50:24 -07001580func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1581 module := &CCLibrary{}
1582 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001583
Colin Cross97ba0732015-03-23 17:50:24 -07001584 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001585}
1586
1587//
1588// Shared libraries
1589//
1590
Colin Cross97ba0732015-03-23 17:50:24 -07001591func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1592 module := &CCLibrary{}
1593 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001594
Colin Cross97ba0732015-03-23 17:50:24 -07001595 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001596}
1597
1598//
1599// Host static library
1600//
1601
Colin Cross97ba0732015-03-23 17:50:24 -07001602func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1603 module := &CCLibrary{}
1604 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001605
Colin Cross97ba0732015-03-23 17:50:24 -07001606 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001607}
1608
1609//
1610// Host Shared libraries
1611//
1612
Colin Cross97ba0732015-03-23 17:50:24 -07001613func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1614 module := &CCLibrary{}
1615 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001616
Colin Cross97ba0732015-03-23 17:50:24 -07001617 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001618}
1619
1620//
1621// Host Binaries
1622//
1623
Colin Cross97ba0732015-03-23 17:50:24 -07001624func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1625 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001626
Colin Cross97ba0732015-03-23 17:50:24 -07001627 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001628}
1629
1630//
Colin Cross1f8f2342015-03-26 16:09:47 -07001631// Host Tests
1632//
1633
1634func CCTestHostFactory() (blueprint.Module, []interface{}) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001635 module := &CCTest{}
Colin Cross1f8f2342015-03-26 16:09:47 -07001636 return NewCCBinary(&module.CCBinary, module, common.HostSupported,
Colin Cross9ffb4f52015-04-24 17:48:09 -07001637 &module.TestProperties)
Colin Cross1f8f2342015-03-26 16:09:47 -07001638}
1639
1640//
Colin Cross2ba19d92015-05-07 15:44:20 -07001641// Host Benchmarks
1642//
1643
1644func CCBenchmarkHostFactory() (blueprint.Module, []interface{}) {
1645 module := &CCBenchmark{}
1646 return NewCCBinary(&module.CCBinary, module, common.HostSupported)
1647}
1648
1649//
Colin Cross3f40fa42015-01-30 17:27:36 -08001650// Device libraries shipped with gcc
1651//
1652
1653type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001654 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001655}
1656
1657func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1658 // toolchain libraries can't have any dependencies
1659 return nil
1660}
1661
Colin Cross0676e2d2015-04-24 17:39:18 -07001662func (*toolchainLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001663 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001664 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001665}
1666
Colin Cross97ba0732015-03-23 17:50:24 -07001667func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001668 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001669
Colin Cross97ba0732015-03-23 17:50:24 -07001670 module.LibraryProperties.BuildStatic = true
1671
Colin Crossfa138792015-04-24 17:31:52 -07001672 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth,
Colin Cross21b9a242015-03-24 14:15:58 -07001673 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001674}
1675
1676func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001677 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001678
1679 libName := ctx.ModuleName() + staticLibraryExtension
1680 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1681
1682 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1683
1684 c.out = outputFile
1685
1686 ctx.CheckbuildFile(outputFile)
1687}
1688
Colin Cross97ba0732015-03-23 17:50:24 -07001689func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001690 // Toolchain libraries do not get installed.
1691}
1692
Dan Albertbe961682015-03-18 23:38:50 -07001693// NDK prebuilt libraries.
1694//
1695// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1696// either (with the exception of the shared STLs, which are installed to the app's directory rather
1697// than to the system image).
1698
1699func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1700 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
Colin Cross1332b002015-04-07 17:11:30 -07001701 ctx.AConfig().SrcDir(), version, toolchain.Name())
Dan Albertbe961682015-03-18 23:38:50 -07001702}
1703
Dan Albertc3144b12015-04-28 18:17:56 -07001704func ndkPrebuiltModuleToPath(ctx common.AndroidModuleContext, toolchain Toolchain,
1705 ext string, version string) string {
1706
1707 // NDK prebuilts are named like: ndk_NAME.EXT.SDK_VERSION.
1708 // We want to translate to just NAME.EXT
1709 name := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1710 dir := getNdkLibDir(ctx, toolchain, version)
1711 return filepath.Join(dir, name+ext)
1712}
1713
1714type ndkPrebuiltObject struct {
1715 ccObject
1716}
1717
1718func (*ndkPrebuiltObject) AndroidDynamicDependencies(
1719 ctx common.AndroidDynamicDependerModuleContext) []string {
1720
1721 // NDK objects can't have any dependencies
1722 return nil
1723}
1724
1725func (*ndkPrebuiltObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1726 // NDK objects can't have any dependencies
1727 return CCDeps{}
1728}
1729
1730func NdkPrebuiltObjectFactory() (blueprint.Module, []interface{}) {
1731 module := &ndkPrebuiltObject{}
1732 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
1733}
1734
1735func (c *ndkPrebuiltObject) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1736 deps CCDeps, objFiles []string) {
1737 // A null build step, but it sets up the output path.
1738 if !strings.HasPrefix(ctx.ModuleName(), "ndk_crt") {
1739 ctx.ModuleErrorf("NDK prebuilts must have an ndk_crt prefixed name")
1740 }
1741
1742 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, objectExtension, c.Properties.Sdk_version)
1743}
1744
1745func (c *ndkPrebuiltObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1746 // Objects do not get installed.
1747}
1748
1749var _ ccObjectProvider = (*ndkPrebuiltObject)(nil)
1750
Dan Albertbe961682015-03-18 23:38:50 -07001751type ndkPrebuiltLibrary struct {
1752 CCLibrary
1753}
1754
1755func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1756 ctx common.AndroidDynamicDependerModuleContext) []string {
1757
1758 // NDK libraries can't have any dependencies
1759 return nil
1760}
1761
Colin Cross0676e2d2015-04-24 17:39:18 -07001762func (*ndkPrebuiltLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Dan Albertbe961682015-03-18 23:38:50 -07001763 // NDK libraries can't have any dependencies
1764 return CCDeps{}
1765}
1766
1767func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1768 module := &ndkPrebuiltLibrary{}
1769 module.LibraryProperties.BuildShared = true
1770 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1771}
1772
1773func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1774 deps CCDeps, objFiles []string) {
1775 // A null build step, but it sets up the output path.
1776 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1777 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1778 }
1779
Colin Crossfa138792015-04-24 17:31:52 -07001780 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001781 c.exportFlags = []string{common.JoinWithPrefix(includeDirs, "-isystem ")}
Dan Albertbe961682015-03-18 23:38:50 -07001782
Dan Albertc3144b12015-04-28 18:17:56 -07001783 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, sharedLibraryExtension,
1784 c.Properties.Sdk_version)
Dan Albertbe961682015-03-18 23:38:50 -07001785}
1786
1787func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc3144b12015-04-28 18:17:56 -07001788 // NDK prebuilt libraries do not get installed.
Dan Albertbe961682015-03-18 23:38:50 -07001789}
1790
1791// The NDK STLs are slightly different from the prebuilt system libraries:
1792// * Are not specific to each platform version.
1793// * The libraries are not in a predictable location for each STL.
1794
1795type ndkPrebuiltStl struct {
1796 ndkPrebuiltLibrary
1797}
1798
1799type ndkPrebuiltStaticStl struct {
1800 ndkPrebuiltStl
1801}
1802
1803type ndkPrebuiltSharedStl struct {
1804 ndkPrebuiltStl
1805}
1806
1807func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1808 module := &ndkPrebuiltSharedStl{}
1809 module.LibraryProperties.BuildShared = true
1810 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1811}
1812
1813func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1814 module := &ndkPrebuiltStaticStl{}
1815 module.LibraryProperties.BuildStatic = true
1816 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1817}
1818
1819func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1820 gccVersion := toolchain.GccVersion()
1821 var libDir string
1822 switch stl {
1823 case "libstlport":
1824 libDir = "cxx-stl/stlport/libs"
1825 case "libc++":
1826 libDir = "cxx-stl/llvm-libc++/libs"
1827 case "libgnustl":
1828 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1829 }
1830
1831 if libDir != "" {
Colin Cross1332b002015-04-07 17:11:30 -07001832 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources"
Dan Albertbe961682015-03-18 23:38:50 -07001833 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1834 }
1835
1836 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1837 return ""
1838}
1839
1840func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1841 deps CCDeps, objFiles []string) {
1842 // A null build step, but it sets up the output path.
1843 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1844 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1845 }
1846
Colin Crossfa138792015-04-24 17:31:52 -07001847 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001848 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Dan Albertbe961682015-03-18 23:38:50 -07001849
1850 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1851 libExt := sharedLibraryExtension
1852 if c.LibraryProperties.BuildStatic {
1853 libExt = staticLibraryExtension
1854 }
1855
1856 stlName := strings.TrimSuffix(libName, "_shared")
1857 stlName = strings.TrimSuffix(stlName, "_static")
1858 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1859 c.out = libDir + "/" + libName + libExt
1860}
1861
Colin Cross3f40fa42015-01-30 17:27:36 -08001862func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001863 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001864 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001865 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001866 modules = mctx.CreateLocalVariations("static", "shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001867 modules[0].(ccLinkedInterface).setStatic(true)
1868 modules[1].(ccLinkedInterface).setStatic(false)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001869 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001870 modules = mctx.CreateLocalVariations("static")
Colin Cross18b6dc52015-04-28 13:20:37 -07001871 modules[0].(ccLinkedInterface).setStatic(true)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001872 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001873 modules = mctx.CreateLocalVariations("shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001874 modules[0].(ccLinkedInterface).setStatic(false)
Colin Cross3f40fa42015-01-30 17:27:36 -08001875 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001876 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001877 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001878
1879 if _, ok := c.(ccLibraryInterface); ok {
1880 reuseFrom := modules[0].(ccLibraryInterface)
1881 for _, m := range modules {
1882 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001883 }
1884 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001885 }
1886}
Colin Cross74d1ec02015-04-28 13:30:13 -07001887
1888// lastUniqueElements returns all unique elements of a slice, keeping the last copy of each
1889// modifies the slice contents in place, and returns a subslice of the original slice
1890func lastUniqueElements(list []string) []string {
1891 totalSkip := 0
1892 for i := len(list) - 1; i >= totalSkip; i-- {
1893 skip := 0
1894 for j := i - 1; j >= totalSkip; j-- {
1895 if list[i] == list[j] {
1896 skip++
1897 } else {
1898 list[j+skip] = list[j]
1899 }
1900 }
1901 totalSkip += skip
1902 }
1903 return list[totalSkip:]
1904}