blob: 2583c86676ca58becc0780ede40733f1c22c3cbc [file] [log] [blame]
Colin Cross5049f022015-03-18 13:28:46 -07001// Copyright 2015 Google Inc. All rights reserved.
Colin Cross3f40fa42015-01-30 17:27:36 -08002//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package cc
16
17// This file contains the module types for compiling C/C++ for Android, and converts the properties
18// into the flags and filenames necessary to pass to the compiler. The final creation of the rules
19// is handled in builder.go
20
21import (
Colin Cross3f40fa42015-01-30 17:27:36 -080022 "fmt"
23 "path/filepath"
24 "strings"
25
Colin Cross97ba0732015-03-23 17:50:24 -070026 "github.com/google/blueprint"
27 "github.com/google/blueprint/pathtools"
28
Colin Cross3f40fa42015-01-30 17:27:36 -080029 "android/soong/common"
Colin Cross5049f022015-03-18 13:28:46 -070030 "android/soong/genrule"
Colin Cross3f40fa42015-01-30 17:27:36 -080031)
32
33type Config interface {
34 SrcDir() string
35 PrebuiltOS() string
36}
37
38var (
39 HostPrebuiltTag = pctx.VariableConfigMethod("HostPrebuiltTag", Config.PrebuiltOS)
40 SrcDir = pctx.VariableConfigMethod("SrcDir", Config.SrcDir)
41
42 LibcRoot = pctx.StaticVariable("LibcRoot", "${SrcDir}/bionic/libc")
43 LibmRoot = pctx.StaticVariable("LibmRoot", "${SrcDir}/bionic/libm")
44)
45
46// Flags used by lots of devices. Putting them in package static variables will save bytes in
47// build.ninja so they aren't repeated for every file
48var (
49 commonGlobalCflags = []string{
50 "-DANDROID",
51 "-fmessage-length=0",
52 "-W",
53 "-Wall",
54 "-Wno-unused",
55 "-Winit-self",
56 "-Wpointer-arith",
57
58 // COMMON_RELEASE_CFLAGS
59 "-DNDEBUG",
60 "-UDEBUG",
61 }
62
63 deviceGlobalCflags = []string{
64 // TARGET_ERROR_FLAGS
65 "-Werror=return-type",
66 "-Werror=non-virtual-dtor",
67 "-Werror=address",
68 "-Werror=sequence-point",
69 }
70
71 hostGlobalCflags = []string{}
72
73 commonGlobalCppflags = []string{
74 "-Wsign-promo",
75 "-std=gnu++11",
76 }
77)
78
79func init() {
80 pctx.StaticVariable("commonGlobalCflags", strings.Join(commonGlobalCflags, " "))
81 pctx.StaticVariable("deviceGlobalCflags", strings.Join(deviceGlobalCflags, " "))
82 pctx.StaticVariable("hostGlobalCflags", strings.Join(hostGlobalCflags, " "))
83
84 pctx.StaticVariable("commonGlobalCppflags", strings.Join(commonGlobalCppflags, " "))
85
86 pctx.StaticVariable("commonClangGlobalCflags",
87 strings.Join(clangFilterUnknownCflags(commonGlobalCflags), " "))
88 pctx.StaticVariable("deviceClangGlobalCflags",
89 strings.Join(clangFilterUnknownCflags(deviceGlobalCflags), " "))
90 pctx.StaticVariable("hostClangGlobalCflags",
91 strings.Join(clangFilterUnknownCflags(hostGlobalCflags), " "))
Tim Kilbournf2948142015-03-11 12:03:03 -070092 pctx.StaticVariable("commonClangGlobalCppflags",
93 strings.Join(clangFilterUnknownCflags(commonGlobalCppflags), " "))
Colin Cross3f40fa42015-01-30 17:27:36 -080094
95 // Everything in this list is a crime against abstraction and dependency tracking.
96 // Do not add anything to this list.
97 pctx.StaticVariable("commonGlobalIncludes", strings.Join([]string{
98 "-isystem ${SrcDir}/system/core/include",
99 "-isystem ${SrcDir}/hardware/libhardware/include",
100 "-isystem ${SrcDir}/hardware/libhardware_legacy/include",
101 "-isystem ${SrcDir}/hardware/ril/include",
102 "-isystem ${SrcDir}/libnativehelper/include",
103 "-isystem ${SrcDir}/frameworks/native/include",
104 "-isystem ${SrcDir}/frameworks/native/opengl/include",
105 "-isystem ${SrcDir}/frameworks/av/include",
106 "-isystem ${SrcDir}/frameworks/base/include",
107 }, " "))
108
109 pctx.StaticVariable("clangPath", "${SrcDir}/prebuilts/clang/${HostPrebuiltTag}/host/3.6/bin/")
110}
111
Colin Cross97ba0732015-03-23 17:50:24 -0700112// ccProperties describes properties used to compile all C or C++ modules
Colin Cross3f40fa42015-01-30 17:27:36 -0800113type ccProperties struct {
114 // srcs: list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
115 Srcs []string `android:"arch_variant,arch_subtract"`
116
117 // cflags: list of module-specific flags that will be used for C and C++ compiles.
118 Cflags []string `android:"arch_variant"`
119
120 // cppflags: list of module-specific flags that will be used for C++ compiles
121 Cppflags []string `android:"arch_variant"`
122
123 // conlyflags: list of module-specific flags that will be used for C compiles
124 Conlyflags []string `android:"arch_variant"`
125
126 // asflags: list of module-specific flags that will be used for .S compiles
127 Asflags []string `android:"arch_variant"`
128
129 // ldflags: list of module-specific flags that will be used for all link steps
130 Ldflags []string `android:"arch_variant"`
131
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700132 // instruction_set: the instruction set architecture to use to compile the C/C++
133 // module.
134 Instruction_set string `android:"arch_variant"`
135
Colin Cross3f40fa42015-01-30 17:27:36 -0800136 // include_dirs: list of directories relative to the root of the source tree that will
137 // be added to the include path using -I.
138 // If possible, don't use this. If adding paths from the current directory use
139 // local_include_dirs, if adding paths from other modules use export_include_dirs in
140 // that module.
141 Include_dirs []string `android:"arch_variant"`
142
143 // local_include_dirs: list of directories relative to the Blueprints file that will
144 // be added to the include path using -I
145 Local_include_dirs []string `android:"arch_variant"`
146
147 // export_include_dirs: list of directories relative to the Blueprints file that will
148 // be added to the include path using -I for any module that links against this module
Dan Albertbe961682015-03-18 23:38:50 -0700149 Export_include_dirs []string `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800150
151 // clang_cflags: list of module-specific flags that will be used for C and C++ compiles when
152 // compiling with clang
153 Clang_cflags []string `android:"arch_variant"`
154
155 // clang_asflags: list of module-specific flags that will be used for .S compiles when
156 // compiling with clang
157 Clang_asflags []string `android:"arch_variant"`
158
159 // system_shared_libs: list of system libraries that will be dynamically linked to
160 // shared library and executable modules. If unset, generally defaults to libc
161 // and libm. Set to [] to prevent linking against libc and libm.
162 System_shared_libs []string
163
164 // whole_static_libs: list of modules whose object files should be linked into this module
165 // in their entirety. For static library modules, all of the .o files from the intermediate
166 // directory of the dependency will be linked into this modules .a file. For a shared library,
167 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
168 Whole_static_libs []string `android:"arch_variant"`
169
170 // static_libs: list of modules that should be statically linked into this module.
171 Static_libs []string `android:"arch_variant"`
172
173 // shared_libs: list of modules that should be dynamically linked into this module.
174 Shared_libs []string `android:"arch_variant"`
175
176 // allow_undefined_symbols: allow the module to contain undefined symbols. By default,
177 // modules cannot contain undefined symbols that are not satisified by their immediate
178 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
179 // This flag should only be necessary for compiling low-level libraries like libc.
180 Allow_undefined_symbols bool
181
182 // nocrt: don't link in crt_begin and crt_end. This flag should only be necessary for
183 // compiling crt or libc.
184 Nocrt bool `android:"arch_variant"`
185
186 // no_default_compiler_flags: don't insert default compiler flags into asflags, cflags,
187 // cppflags, conlyflags, ldflags, or include_dirs
188 No_default_compiler_flags bool
189
190 // clang: compile module with clang instead of gcc
191 Clang bool `android:"arch_variant"`
192
193 // rtti: pass -frtti instead of -fno-rtti
194 Rtti bool
195
196 // host_ldlibs: -l arguments to pass to linker for host-provided shared libraries
197 Host_ldlibs []string `android:"arch_variant"`
198
199 // stl: select the STL library to use. Possible values are "libc++", "libc++_static",
200 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
201 // default
202 Stl string
203
204 // Set for combined shared/static libraries to prevent compiling object files a second time
205 SkipCompileObjs bool `blueprint:"mutated"`
Colin Crossaf19a292015-03-18 12:07:10 -0700206
207 Debug struct {
208 Cflags []string `android:"arch_variant"`
209 } `android:"arch_variant"`
210 Release struct {
211 Cflags []string `android:"arch_variant"`
212 } `android:"arch_variant"`
Colin Crossefd8e482015-03-18 17:17:35 -0700213
214 // Minimum sdk version supported when compiling against the ndk
215 Sdk_version string
Colin Cross3f40fa42015-01-30 17:27:36 -0800216}
217
218type unusedProperties struct {
219 Asan bool
220 Native_coverage bool
221 Strip string
222 Tags []string
223 Required []string
224}
225
226// Building C/C++ code is handled by objects that satisfy this interface via composition
Colin Cross97ba0732015-03-23 17:50:24 -0700227type CCModuleType interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800228 common.AndroidModule
229
Colin Cross21b9a242015-03-24 14:15:58 -0700230 // Modify the ccFlags
231 Flags(common.AndroidModuleContext, CCFlags) CCFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800232
Colin Cross21b9a242015-03-24 14:15:58 -0700233 // Return list of dependency names for use in AndroidDynamicDependencies and in depsToPaths
234 DepNames(common.AndroidBaseContext, CCDeps) CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800235
236 // Compile objects into final module
Colin Cross97ba0732015-03-23 17:50:24 -0700237 compileModule(common.AndroidModuleContext, CCFlags, CCDeps, []string)
Colin Cross3f40fa42015-01-30 17:27:36 -0800238
Dan Albertc403f7c2015-03-18 14:01:18 -0700239 // Install the built module.
Colin Cross97ba0732015-03-23 17:50:24 -0700240 installModule(common.AndroidModuleContext, CCFlags)
Dan Albertc403f7c2015-03-18 14:01:18 -0700241
Colin Cross3f40fa42015-01-30 17:27:36 -0800242 // Return the output file (.o, .a or .so) for use by other modules
243 outputFile() string
244}
245
Colin Cross97ba0732015-03-23 17:50:24 -0700246type CCDeps struct {
247 StaticLibs, SharedLibs, LateStaticLibs, WholeStaticLibs, ObjFiles, IncludeDirs []string
Colin Crossc472d572015-03-17 15:06:21 -0700248
Colin Cross21b9a242015-03-24 14:15:58 -0700249 WholeStaticLibObjFiles []string
250
Colin Cross97ba0732015-03-23 17:50:24 -0700251 CrtBegin, CrtEnd string
Colin Crossc472d572015-03-17 15:06:21 -0700252}
253
Colin Cross97ba0732015-03-23 17:50:24 -0700254type CCFlags struct {
255 GlobalFlags []string
256 AsFlags []string
257 CFlags []string
258 ConlyFlags []string
259 CppFlags []string
260 LdFlags []string
261 LdLibs []string
262 IncludeDirs []string
263 Nocrt bool
264 Toolchain Toolchain
265 Clang bool
Colin Crossc472d572015-03-17 15:06:21 -0700266}
267
268// ccBase contains the properties and members used by all C/C++ module types, and implements
269// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
270// and uses a ccModuleType interface to that struct to create the build steps.
271type ccBase struct {
272 common.AndroidModuleBase
Colin Cross97ba0732015-03-23 17:50:24 -0700273 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700274
275 properties ccProperties
276 unused unusedProperties
277
278 installPath string
279}
280
Colin Cross97ba0732015-03-23 17:50:24 -0700281func newCCBase(base *ccBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700282 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
283
284 base.module = module
285
286 props = append(props, &base.properties, &base.unused)
287
Colin Cross5049f022015-03-18 13:28:46 -0700288 return common.InitAndroidArchModule(module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700289}
290
Colin Cross3f40fa42015-01-30 17:27:36 -0800291func (c *ccBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
292 toolchain := c.findToolchain(ctx)
293 if ctx.Failed() {
294 return
295 }
296
Colin Cross21b9a242015-03-24 14:15:58 -0700297 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800298 if ctx.Failed() {
299 return
300 }
301
Colin Cross21b9a242015-03-24 14:15:58 -0700302 depNames := c.module.DepNames(ctx, CCDeps{})
Colin Cross3f40fa42015-01-30 17:27:36 -0800303 if ctx.Failed() {
304 return
305 }
306
Colin Cross21b9a242015-03-24 14:15:58 -0700307 deps := c.depsToPaths(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800308 if ctx.Failed() {
309 return
310 }
311
Colin Cross97ba0732015-03-23 17:50:24 -0700312 flags.IncludeDirs = append(flags.IncludeDirs, deps.IncludeDirs...)
Colin Crossed9f8682015-03-18 17:17:35 -0700313
Colin Cross3f40fa42015-01-30 17:27:36 -0800314 objFiles := c.compileObjs(ctx, flags, deps)
315 if ctx.Failed() {
316 return
317 }
318
Colin Cross5049f022015-03-18 13:28:46 -0700319 generatedObjFiles := c.compileGeneratedObjs(ctx, flags, deps)
320 if ctx.Failed() {
321 return
322 }
323
324 objFiles = append(objFiles, generatedObjFiles...)
325
Colin Cross3f40fa42015-01-30 17:27:36 -0800326 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
327 if ctx.Failed() {
328 return
329 }
Dan Albertc403f7c2015-03-18 14:01:18 -0700330
331 c.ccModuleType().installModule(ctx, flags)
332 if ctx.Failed() {
333 return
334 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800335}
336
Colin Cross97ba0732015-03-23 17:50:24 -0700337func (c *ccBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800338 return c.module
339}
340
341var _ common.AndroidDynamicDepender = (*ccBase)(nil)
342
Colin Cross97ba0732015-03-23 17:50:24 -0700343func (c *ccBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800344 arch := ctx.Arch()
345 factory := toolchainFactories[arch.HostOrDevice][arch.ArchType]
346 if factory == nil {
347 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
348 arch.HostOrDevice.String(), arch.String()))
349 }
350 return factory(arch.ArchVariant, arch.CpuVariant)
351}
352
Dan Albertbe961682015-03-18 23:38:50 -0700353func addNdkStlDepNames(ctx common.AndroidBaseContext, stl string, depNames CCDeps) CCDeps {
354 if stl == "ndk_system" {
355 // TODO: Make a system STL prebuilt for the NDK.
356 // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have
357 // its own includes. The includes are handled in ccBase.Flags().
358 return depNames
359 }
360
361 if strings.HasSuffix(stl, "_static") {
362 depNames.StaticLibs = append(depNames.StaticLibs, stl)
363 } else {
364 depNames.SharedLibs = append(depNames.SharedLibs, stl)
365 }
366
367 return depNames
368}
369
Colin Cross21b9a242015-03-24 14:15:58 -0700370func (c *ccBase) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
371 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.properties.Whole_static_libs...)
372 depNames.StaticLibs = append(depNames.StaticLibs, c.properties.Static_libs...)
373 depNames.SharedLibs = append(depNames.SharedLibs, c.properties.Shared_libs...)
374
375 stl := c.stl(ctx)
376 if ctx.Failed() {
377 return depNames
378 }
379
380 switch stl {
381 case "libc++", "libstdc++":
382 depNames.SharedLibs = append(depNames.SharedLibs, stl)
383 case "libc++_static":
384 depNames.StaticLibs = append(depNames.StaticLibs, stl)
385 case "stlport":
386 depNames.SharedLibs = append(depNames.SharedLibs, "libstdc++", "libstlport")
387 case "stlport_static":
388 depNames.StaticLibs = append(depNames.StaticLibs, "libstdc++", "libstlport_static")
Dan Albertbe961682015-03-18 23:38:50 -0700389 case "":
390 // None or error.
391 default:
392 if !strings.HasPrefix(stl, "ndk_") {
393 panic("unexpected case")
394 }
395
396 depNames = addNdkStlDepNames(ctx, stl, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -0700397 }
398
399 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800400}
401
402func (c *ccBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
Colin Cross21b9a242015-03-24 14:15:58 -0700403 depNames := CCDeps{}
404 depNames = c.module.DepNames(ctx, depNames)
405 staticLibs := depNames.WholeStaticLibs
406 staticLibs = append(staticLibs, depNames.StaticLibs...)
407 staticLibs = append(staticLibs, depNames.LateStaticLibs...)
408 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800409
Colin Cross21b9a242015-03-24 14:15:58 -0700410 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, depNames.SharedLibs...)
411
412 ret := append([]string(nil), depNames.ObjFiles...)
413 if depNames.CrtBegin != "" {
414 ret = append(ret, depNames.CrtBegin)
415 }
416 if depNames.CrtEnd != "" {
417 ret = append(ret, depNames.CrtEnd)
418 }
419
420 return ret
Colin Cross3f40fa42015-01-30 17:27:36 -0800421}
422
423// Create a ccFlags struct that collects the compile flags from global values,
424// per-target values, module type values, and per-module Blueprints properties
Colin Cross21b9a242015-03-24 14:15:58 -0700425func (c *ccBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700426 flags := CCFlags{
427 CFlags: c.properties.Cflags,
428 CppFlags: c.properties.Cppflags,
429 ConlyFlags: c.properties.Conlyflags,
430 LdFlags: c.properties.Ldflags,
431 AsFlags: c.properties.Asflags,
432 Nocrt: c.properties.Nocrt,
433 Toolchain: toolchain,
434 Clang: c.properties.Clang,
Colin Cross3f40fa42015-01-30 17:27:36 -0800435 }
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700436 instructionSet := c.properties.Instruction_set
437 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
438 if err != nil {
439 ctx.ModuleErrorf("%s", err)
440 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800441
Colin Crossaf19a292015-03-18 12:07:10 -0700442 // TODO: debug
Colin Cross97ba0732015-03-23 17:50:24 -0700443 flags.CFlags = append(flags.CFlags, c.properties.Release.Cflags...)
Colin Crossaf19a292015-03-18 12:07:10 -0700444
Colin Crossf6566ed2015-03-24 11:13:38 -0700445 if ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800446 // TODO: allow per-module clang disable for host
Colin Cross97ba0732015-03-23 17:50:24 -0700447 flags.Clang = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800448 }
449
Colin Cross97ba0732015-03-23 17:50:24 -0700450 if flags.Clang {
451 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
452 flags.CFlags = append(flags.CFlags, c.properties.Clang_cflags...)
453 flags.AsFlags = append(flags.AsFlags, c.properties.Clang_asflags...)
454 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
455 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
456 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800457
Colin Cross97ba0732015-03-23 17:50:24 -0700458 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
459 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700460 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700461 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700462 }
463
Colin Cross3f40fa42015-01-30 17:27:36 -0800464 target := "-target " + toolchain.ClangTriple()
465 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
466
Colin Cross97ba0732015-03-23 17:50:24 -0700467 flags.CFlags = append(flags.CFlags, target, gccPrefix)
468 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
469 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800470
Colin Crossf6566ed2015-03-24 11:13:38 -0700471 if ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800472 gccToolchain := "--gcc-toolchain=" + toolchain.GccRoot()
473 sysroot := "--sysroot=" + filepath.Join(toolchain.GccRoot(), "sysroot")
474
475 // TODO: also need more -B, -L flags to make host builds hermetic
Colin Cross97ba0732015-03-23 17:50:24 -0700476 flags.CFlags = append(flags.CFlags, gccToolchain, sysroot)
477 flags.AsFlags = append(flags.AsFlags, gccToolchain, sysroot)
478 flags.LdFlags = append(flags.LdFlags, gccToolchain, sysroot)
Colin Cross3f40fa42015-01-30 17:27:36 -0800479 }
480 }
481
Colin Cross97ba0732015-03-23 17:50:24 -0700482 flags.IncludeDirs = pathtools.PrefixPaths(c.properties.Include_dirs, ctx.Config().(Config).SrcDir())
Colin Cross3f40fa42015-01-30 17:27:36 -0800483 localIncludeDirs := pathtools.PrefixPaths(c.properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross97ba0732015-03-23 17:50:24 -0700484 flags.IncludeDirs = append(flags.IncludeDirs, localIncludeDirs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800485
486 if !c.properties.No_default_compiler_flags {
Colin Cross97ba0732015-03-23 17:50:24 -0700487 flags.IncludeDirs = append(flags.IncludeDirs, []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800488 common.ModuleSrcDir(ctx),
489 common.ModuleOutDir(ctx),
490 common.ModuleGenDir(ctx),
491 }...)
492
Colin Crossefd8e482015-03-18 17:17:35 -0700493 if c.properties.Sdk_version == "" {
Colin Cross97ba0732015-03-23 17:50:24 -0700494 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/libnativehelper/include/nativehelper")
Colin Crossefd8e482015-03-18 17:17:35 -0700495 }
496
Colin Crossf6566ed2015-03-24 11:13:38 -0700497 if ctx.Device() && !c.properties.Allow_undefined_symbols {
Colin Cross97ba0732015-03-23 17:50:24 -0700498 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800499 }
500
Colin Cross97ba0732015-03-23 17:50:24 -0700501 if flags.Clang {
502 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
503 flags.GlobalFlags = []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800504 "${commonGlobalIncludes}",
505 toolchain.IncludeFlags(),
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700506 instructionSetFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800507 toolchain.ClangCflags(),
508 "${commonClangGlobalCflags}",
Colin Crossf6566ed2015-03-24 11:13:38 -0700509 fmt.Sprintf("${%sClangGlobalCflags}", ctx.Arch().HostOrDevice),
Colin Cross3f40fa42015-01-30 17:27:36 -0800510 }
511 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700512 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
513 flags.GlobalFlags = []string{
Colin Cross3f40fa42015-01-30 17:27:36 -0800514 "${commonGlobalIncludes}",
515 toolchain.IncludeFlags(),
Tim Kilbourn1a9bf262015-03-18 12:28:32 -0700516 instructionSetFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800517 toolchain.Cflags(),
518 "${commonGlobalCflags}",
Colin Crossf6566ed2015-03-24 11:13:38 -0700519 fmt.Sprintf("${%sGlobalCflags}", ctx.Arch().HostOrDevice),
Colin Cross3f40fa42015-01-30 17:27:36 -0800520 }
521 }
522
Colin Crossf6566ed2015-03-24 11:13:38 -0700523 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -0700524 flags.LdFlags = append(flags.LdFlags, c.properties.Host_ldlibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800525 }
526
Colin Crossf6566ed2015-03-24 11:13:38 -0700527 if ctx.Device() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800528 if c.properties.Rtti {
Colin Cross97ba0732015-03-23 17:50:24 -0700529 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800530 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700531 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800532 }
533 }
534
Colin Cross97ba0732015-03-23 17:50:24 -0700535 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800536
Colin Cross97ba0732015-03-23 17:50:24 -0700537 if flags.Clang {
538 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
539 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800540 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700541 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
542 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800543 }
544 }
545
Colin Cross21b9a242015-03-24 14:15:58 -0700546 flags = c.ccModuleType().Flags(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800547
548 // Optimization to reduce size of build.ninja
549 // Replace the long list of flags for each file with a module-local variable
Colin Cross97ba0732015-03-23 17:50:24 -0700550 ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " "))
551 ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " "))
552 ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " "))
553 flags.CFlags = []string{"$cflags"}
554 flags.CppFlags = []string{"$cppflags"}
555 flags.AsFlags = []string{"$asflags"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800556
557 return flags
558}
559
Colin Cross21b9a242015-03-24 14:15:58 -0700560func (c *ccBase) stl(ctx common.AndroidBaseContext) string {
Dan Albertbe961682015-03-18 23:38:50 -0700561 if c.properties.Sdk_version != "" {
562 stl := c.properties.Stl
563 if stl == "" {
564 return "ndk_system"
565 }
566 return "ndk_lib" + stl
567 }
568
Colin Cross21b9a242015-03-24 14:15:58 -0700569 switch c.properties.Stl {
570 case "libc++", "libc++_static",
571 "stlport", "stlport_static",
572 "libstdc++":
573 return c.properties.Stl
574 case "none":
575 return ""
576 case "":
577 return "libc++" // TODO: mingw needs libstdc++
Colin Cross21b9a242015-03-24 14:15:58 -0700578 default:
579 ctx.ModuleErrorf("stl: %q is not a supported STL", c.properties.Stl)
580 return ""
Colin Cross3f40fa42015-01-30 17:27:36 -0800581 }
Colin Cross21b9a242015-03-24 14:15:58 -0700582}
583
584func (c *ccBase) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
585 stl := c.stl(ctx)
586 if ctx.Failed() {
587 return flags
588 }
589
590 switch stl {
591 case "libc++", "libc++_static":
592 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
593 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/external/libcxx/include")
594 if ctx.Host() {
595 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
596 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
597 flags.LdLibs = append(flags.LdLibs, "-lc", "-lm", "-lpthread")
598 }
599 case "stlport", "stlport_static":
600 if ctx.Device() {
601 flags.IncludeDirs = append(flags.IncludeDirs,
602 "${SrcDir}/external/stlport/stlport",
603 "${SrcDir}/bionic/libstdc++/include",
604 "${SrcDir}/bionic")
605 }
Colin Cross21b9a242015-03-24 14:15:58 -0700606 case "libstdc++":
607 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
608 // tree is in good enough shape to not need it.
609 // Host builds will use GNU libstdc++.
610 if ctx.Device() {
611 flags.IncludeDirs = append(flags.IncludeDirs, "${SrcDir}/bionic/libstdc++/include")
612 }
Dan Albertbe961682015-03-18 23:38:50 -0700613 case "ndk_system":
614 ndkSrcRoot := ctx.Config().(Config).SrcDir() + "/prebuilts/ndk/current/sources/"
615 flags.IncludeDirs = append(flags.IncludeDirs, ndkSrcRoot + "cxx-stl/system/include")
616 case "ndk_c++_shared", "ndk_c++_static":
617 // TODO(danalbert): This really shouldn't be here...
618 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
Colin Cross21b9a242015-03-24 14:15:58 -0700619 case "":
Dan Albertbe961682015-03-18 23:38:50 -0700620 // None or error.
Colin Cross21b9a242015-03-24 14:15:58 -0700621 if ctx.Host() {
622 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
623 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
624 flags.LdLibs = append(flags.LdLibs, "-lc", "-lm")
625 }
Colin Cross21b9a242015-03-24 14:15:58 -0700626 }
627
Colin Cross3f40fa42015-01-30 17:27:36 -0800628 return flags
629}
630
631// Compile a list of source files into objects a specified subdirectory
Colin Cross97ba0732015-03-23 17:50:24 -0700632func (c *ccBase) customCompileObjs(ctx common.AndroidModuleContext, flags CCFlags,
633 deps CCDeps, subdir string, srcFiles []string) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800634
635 srcFiles = pathtools.PrefixPaths(srcFiles, common.ModuleSrcDir(ctx))
636 srcFiles = common.ExpandGlobs(ctx, srcFiles)
637
638 return TransformSourceToObj(ctx, subdir, srcFiles, ccFlagsToBuilderFlags(flags))
639}
640
641// Compile files listed in c.properties.Srcs into objects
Colin Cross97ba0732015-03-23 17:50:24 -0700642func (c *ccBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags,
643 deps CCDeps) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800644
645 if c.properties.SkipCompileObjs {
646 return nil
647 }
648
649 return c.customCompileObjs(ctx, flags, deps, "", c.properties.Srcs)
650}
651
Colin Cross5049f022015-03-18 13:28:46 -0700652// Compile generated source files from dependencies
Colin Cross97ba0732015-03-23 17:50:24 -0700653func (c *ccBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags,
654 deps CCDeps) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700655 var srcs []string
656
657 if c.properties.SkipCompileObjs {
658 return nil
659 }
660
661 ctx.VisitDirectDeps(func(module blueprint.Module) {
662 if gen, ok := module.(genrule.SourceFileGenerator); ok {
663 srcs = append(srcs, gen.GeneratedSourceFiles()...)
664 }
665 })
666
667 if len(srcs) == 0 {
668 return nil
669 }
670
671 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags))
672}
673
Colin Cross3f40fa42015-01-30 17:27:36 -0800674func (c *ccBase) outputFile() string {
675 return ""
676}
677
Colin Cross21b9a242015-03-24 14:15:58 -0700678func (c *ccBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800679 names []string) (modules []common.AndroidModule,
680 outputFiles []string, exportedIncludeDirs []string) {
681
682 for _, n := range names {
683 found := false
684 ctx.VisitDirectDeps(func(m blueprint.Module) {
685 otherName := ctx.OtherModuleName(m)
686 if otherName != n {
687 return
688 }
689
Colin Cross97ba0732015-03-23 17:50:24 -0700690 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800691 if a.Disabled() {
692 // If a cc_library host+device module depends on a library that exists as both
693 // cc_library_shared and cc_library_host_shared, it will end up with two
694 // dependencies with the same name, one of which is marked disabled for each
695 // of host and device. Ignore the disabled one.
696 return
697 }
698 if a.HostOrDevice() != ctx.Arch().HostOrDevice {
699 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
700 otherName)
701 return
702 }
703
704 if outputFile := a.outputFile(); outputFile != "" {
705 if found {
706 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
707 return
708 }
709 outputFiles = append(outputFiles, outputFile)
710 modules = append(modules, a)
711 if i, ok := a.(ccExportedIncludeDirsProducer); ok {
712 exportedIncludeDirs = append(exportedIncludeDirs, i.exportedIncludeDirs()...)
713 }
714 found = true
715 } else {
716 ctx.ModuleErrorf("module %q missing output file", otherName)
717 return
718 }
719 } else {
720 ctx.ModuleErrorf("module %q not an android module", otherName)
721 return
722 }
723 })
724 if !found {
725 ctx.ModuleErrorf("unsatisified dependency on %q", n)
726 }
727 }
728
729 return modules, outputFiles, exportedIncludeDirs
730}
731
Colin Cross21b9a242015-03-24 14:15:58 -0700732// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
733// containing paths
734func (c *ccBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
735 var depPaths CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800736 var newIncludeDirs []string
737
Colin Cross21b9a242015-03-24 14:15:58 -0700738 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800739
Colin Cross21b9a242015-03-24 14:15:58 -0700740 wholeStaticLibModules, depPaths.WholeStaticLibs, newIncludeDirs =
741 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
742 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800743
Colin Cross21b9a242015-03-24 14:15:58 -0700744 for _, m := range wholeStaticLibModules {
745 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
746 depPaths.WholeStaticLibObjFiles =
747 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
748 } else {
749 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
750 }
751 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800752
Colin Cross21b9a242015-03-24 14:15:58 -0700753 _, depPaths.StaticLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.StaticLibs)
754 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
755
756 _, depPaths.LateStaticLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
757 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
758
759 _, depPaths.SharedLibs, newIncludeDirs = c.depsToPathsFromList(ctx, depNames.SharedLibs)
760 depPaths.IncludeDirs = append(depPaths.IncludeDirs, newIncludeDirs...)
761
762 ctx.VisitDirectDeps(func(m blueprint.Module) {
763 if obj, ok := m.(*ccObject); ok {
764 otherName := ctx.OtherModuleName(m)
765 if otherName == depNames.CrtBegin {
766 if !c.properties.Nocrt {
767 depPaths.CrtBegin = obj.outputFile()
768 }
769 } else if otherName == depNames.CrtEnd {
770 if !c.properties.Nocrt {
771 depPaths.CrtEnd = obj.outputFile()
772 }
773 } else {
774 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.outputFile())
775 }
776 }
777 })
778
779 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800780}
781
782// ccDynamic contains the properties and members used by shared libraries and dynamic executables
783type ccDynamic struct {
784 ccBase
785}
786
Colin Cross97ba0732015-03-23 17:50:24 -0700787func newCCDynamic(dynamic *ccDynamic, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700788 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
789
790 dynamic.properties.System_shared_libs = []string{defaultSystemSharedLibraries}
791
792 return newCCBase(&dynamic.ccBase, module, hod, multilib, props...)
793}
794
Colin Cross3f40fa42015-01-30 17:27:36 -0800795const defaultSystemSharedLibraries = "__default__"
796
Colin Crossf6566ed2015-03-24 11:13:38 -0700797func (c *ccDynamic) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800798 if len(c.properties.System_shared_libs) == 1 &&
799 c.properties.System_shared_libs[0] == defaultSystemSharedLibraries {
800
Colin Crossf6566ed2015-03-24 11:13:38 -0700801 if ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800802 return []string{}
Dan Albertbe961682015-03-18 23:38:50 -0700803 } else if c.properties.Sdk_version != "" {
804 version := c.properties.Sdk_version
805 libs := []string{
806 "ndk_libc." + version,
807 "ndk_libm." + version,
808 }
809
810 if c.properties.Sdk_version != "" && c.stl(ctx) == "ndk_system" {
811 libs = append([]string{"libstdc++"}, libs...)
812 }
813 return libs
Colin Cross3f40fa42015-01-30 17:27:36 -0800814 } else {
815 return []string{"libc", "libm"}
816 }
817 }
818 return c.properties.System_shared_libs
819}
820
Colin Cross21b9a242015-03-24 14:15:58 -0700821func (c *ccDynamic) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
822 depNames = c.ccBase.DepNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800823
Colin Crossf6566ed2015-03-24 11:13:38 -0700824 if ctx.Device() {
Colin Cross21b9a242015-03-24 14:15:58 -0700825 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
Colin Cross77b00fa2015-03-16 16:15:49 -0700826 // libgcc and libatomic have to be last on the command line
Colin Cross21b9a242015-03-24 14:15:58 -0700827 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic", "libgcc")
828 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800829 }
830
Colin Cross21b9a242015-03-24 14:15:58 -0700831 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800832}
833
834type ccExportedIncludeDirsProducer interface {
835 exportedIncludeDirs() []string
836}
837
838//
839// Combined static+shared libraries
840//
841
Colin Cross97ba0732015-03-23 17:50:24 -0700842type CCLibrary struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800843 ccDynamic
844
Colin Cross97ba0732015-03-23 17:50:24 -0700845 primary *CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -0800846 primaryObjFiles []string
847 objFiles []string
848 exportIncludeDirs []string
849 out string
850
Colin Cross97ba0732015-03-23 17:50:24 -0700851 LibraryProperties struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800852 BuildStatic bool `blueprint:"mutated"`
853 BuildShared bool `blueprint:"mutated"`
854 IsShared bool `blueprint:"mutated"`
855 IsStatic bool `blueprint:"mutated"`
856
857 Static struct {
858 Srcs []string `android:"arch_variant"`
859 Cflags []string `android:"arch_variant"`
860 } `android:"arch_variant"`
861 Shared struct {
862 Srcs []string `android:"arch_variant"`
863 Cflags []string `android:"arch_variant"`
864 } `android:"arch_variant"`
865 }
866}
867
Colin Cross97ba0732015-03-23 17:50:24 -0700868type ccLibraryInterface interface {
869 ccLibrary() *CCLibrary
870 static() bool
871 shared() bool
872 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -0700873}
874
Colin Cross97ba0732015-03-23 17:50:24 -0700875func (c *CCLibrary) ccLibrary() *CCLibrary {
876 return c
Colin Cross3f40fa42015-01-30 17:27:36 -0800877}
878
Colin Cross97ba0732015-03-23 17:50:24 -0700879func (c *CCLibrary) static() bool {
880 return c.LibraryProperties.IsStatic
881}
882
883func (c *CCLibrary) shared() bool {
884 return c.LibraryProperties.IsShared
885}
886
887func NewCCLibrary(library *CCLibrary, module CCModuleType,
888 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
889
890 return newCCDynamic(&library.ccDynamic, module, hod, common.MultilibBoth,
891 &library.LibraryProperties)
892}
893
894func CCLibraryFactory() (blueprint.Module, []interface{}) {
895 module := &CCLibrary{}
896
897 module.LibraryProperties.BuildShared = true
898 module.LibraryProperties.BuildStatic = true
899
900 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
901}
902
Colin Cross21b9a242015-03-24 14:15:58 -0700903func (c *CCLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
904 if c.shared() {
905 depNames = c.ccDynamic.DepNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -0700906 if ctx.Device() {
Colin Cross21b9a242015-03-24 14:15:58 -0700907 depNames.CrtBegin = "crtbegin_so"
908 depNames.CrtEnd = "crtend_so"
Colin Cross3f40fa42015-01-30 17:27:36 -0800909 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800910 } else {
Colin Cross21b9a242015-03-24 14:15:58 -0700911 depNames = c.ccBase.DepNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800912 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800913
Colin Cross21b9a242015-03-24 14:15:58 -0700914 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800915}
916
Colin Cross97ba0732015-03-23 17:50:24 -0700917func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800918 return c.out
919}
920
Colin Cross97ba0732015-03-23 17:50:24 -0700921func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800922 return c.objFiles
923}
924
Colin Cross97ba0732015-03-23 17:50:24 -0700925func (c *CCLibrary) exportedIncludeDirs() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800926 return c.exportIncludeDirs
927}
928
Colin Cross21b9a242015-03-24 14:15:58 -0700929func (c *CCLibrary) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
930 flags = c.ccDynamic.Flags(ctx, flags)
931
Colin Cross97ba0732015-03-23 17:50:24 -0700932 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -0800933
Colin Cross97ba0732015-03-23 17:50:24 -0700934 if c.LibraryProperties.IsShared {
Colin Cross3f40fa42015-01-30 17:27:36 -0800935 libName := ctx.ModuleName()
936 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
937 sharedFlag := "-Wl,-shared"
Colin Crossf6566ed2015-03-24 11:13:38 -0700938 if c.properties.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800939 sharedFlag = "-shared"
940 }
Colin Crossf6566ed2015-03-24 11:13:38 -0700941 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700942 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -0800943 }
Colin Cross97ba0732015-03-23 17:50:24 -0700944
945 flags.LdFlags = append(flags.LdFlags,
946 "-Wl,--gc-sections",
947 sharedFlag,
948 "-Wl,-soname,"+libName+sharedLibraryExtension,
949 )
Colin Cross3f40fa42015-01-30 17:27:36 -0800950 }
Colin Cross97ba0732015-03-23 17:50:24 -0700951
952 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -0800953}
954
Colin Cross97ba0732015-03-23 17:50:24 -0700955func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
956 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800957
958 staticFlags := flags
Colin Cross97ba0732015-03-23 17:50:24 -0700959 staticFlags.CFlags = append(staticFlags.CFlags, c.LibraryProperties.Static.Cflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800960 objFilesStatic := c.customCompileObjs(ctx, staticFlags, deps, common.DeviceStaticLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -0700961 c.LibraryProperties.Static.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800962
963 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -0700964 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800965
966 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
967
968 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
969
970 c.objFiles = objFiles
971 c.out = outputFile
972 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
973 common.ModuleSrcDir(ctx))
974
975 ctx.CheckbuildFile(outputFile)
976}
977
Colin Cross97ba0732015-03-23 17:50:24 -0700978func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
979 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800980
981 sharedFlags := flags
Colin Cross97ba0732015-03-23 17:50:24 -0700982 sharedFlags.CFlags = append(sharedFlags.CFlags, c.LibraryProperties.Shared.Cflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800983 objFilesShared := c.customCompileObjs(ctx, sharedFlags, deps, common.DeviceSharedLibrary,
Colin Cross97ba0732015-03-23 17:50:24 -0700984 c.LibraryProperties.Shared.Srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800985
986 objFiles = append(objFiles, objFilesShared...)
987
988 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
989
Colin Cross97ba0732015-03-23 17:50:24 -0700990 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
991 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd,
Colin Cross77b00fa2015-03-16 16:15:49 -0700992 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -0800993
994 c.out = outputFile
995 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
996 common.ModuleSrcDir(ctx))
Colin Cross3f40fa42015-01-30 17:27:36 -0800997}
998
Colin Cross97ba0732015-03-23 17:50:24 -0700999func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1000 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001001
1002 // Reuse the object files from the matching static library if it exists
1003 if c.primary == c {
1004 c.primaryObjFiles = objFiles
1005 } else {
1006 objFiles = append([]string(nil), c.primary.primaryObjFiles...)
1007 }
1008
Colin Cross97ba0732015-03-23 17:50:24 -07001009 if c.LibraryProperties.IsStatic {
Colin Cross3f40fa42015-01-30 17:27:36 -08001010 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1011 } else {
1012 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1013 }
1014}
1015
Colin Cross97ba0732015-03-23 17:50:24 -07001016func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001017 // Static libraries do not get installed.
1018}
1019
Colin Cross97ba0732015-03-23 17:50:24 -07001020func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001021 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001022 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001023 installDir = "lib64"
1024 }
1025
1026 ctx.InstallFile(installDir, c.out)
1027}
1028
Colin Cross97ba0732015-03-23 17:50:24 -07001029func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1030 if c.LibraryProperties.IsStatic {
Dan Albertc403f7c2015-03-18 14:01:18 -07001031 c.installStaticLibrary(ctx, flags)
1032 } else {
1033 c.installSharedLibrary(ctx, flags)
1034 }
1035}
1036
Colin Cross3f40fa42015-01-30 17:27:36 -08001037//
1038// Objects (for crt*.o)
1039//
1040
1041type ccObject struct {
1042 ccBase
1043 out string
1044}
1045
Colin Cross97ba0732015-03-23 17:50:24 -07001046func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001047 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001048
Colin Crossc472d572015-03-17 15:06:21 -07001049 return newCCBase(&module.ccBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001050}
1051
1052func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1053 // object files can't have any dynamic dependencies
1054 return nil
1055}
1056
Colin Cross21b9a242015-03-24 14:15:58 -07001057func (*ccObject) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1058 // object files can't have any dynamic dependencies
1059 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001060}
1061
1062func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001063 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001064
Colin Cross97ba0732015-03-23 17:50:24 -07001065 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001066
1067 var outputFile string
1068 if len(objFiles) == 1 {
1069 outputFile = objFiles[0]
1070 } else {
1071 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+".o")
1072 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1073 }
1074
1075 c.out = outputFile
1076
1077 ctx.CheckbuildFile(outputFile)
1078}
1079
Colin Cross97ba0732015-03-23 17:50:24 -07001080func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001081 // Object files do not get installed.
1082}
1083
Colin Cross3f40fa42015-01-30 17:27:36 -08001084func (c *ccObject) outputFile() string {
1085 return c.out
1086}
1087
1088//
1089// Executables
1090//
1091
Colin Cross97ba0732015-03-23 17:50:24 -07001092type CCBinary struct {
Colin Cross3f40fa42015-01-30 17:27:36 -08001093 ccDynamic
Dan Albertc403f7c2015-03-18 14:01:18 -07001094 out string
Colin Cross97ba0732015-03-23 17:50:24 -07001095 BinaryProperties struct {
1096 // static_executable: compile executable with -static
1097 Static_executable bool
1098
1099 // stem: set the name of the output
1100 Stem string `android:"arch_variant"`
1101
1102 // prefix_symbols: if set, add an extra objcopy --prefix-symbols= step
1103 Prefix_symbols string
1104 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001105}
1106
Colin Cross97ba0732015-03-23 17:50:24 -07001107func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
1108 if c.BinaryProperties.Stem != "" {
1109 return c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001110 }
1111 return ctx.ModuleName()
1112}
1113
Colin Cross21b9a242015-03-24 14:15:58 -07001114func (c *CCBinary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1115 depNames = c.ccDynamic.DepNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001116 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001117 if c.BinaryProperties.Static_executable {
Colin Cross21b9a242015-03-24 14:15:58 -07001118 depNames.CrtBegin = "crtbegin_static"
Colin Cross3f40fa42015-01-30 17:27:36 -08001119 } else {
Colin Cross21b9a242015-03-24 14:15:58 -07001120 depNames.CrtBegin = "crtbegin_dynamic"
Colin Cross3f40fa42015-01-30 17:27:36 -08001121 }
Colin Cross21b9a242015-03-24 14:15:58 -07001122 depNames.CrtEnd = "crtend_android"
Colin Cross3f40fa42015-01-30 17:27:36 -08001123 }
Colin Cross21b9a242015-03-24 14:15:58 -07001124 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001125}
1126
Colin Cross97ba0732015-03-23 17:50:24 -07001127func NewCCBinary(binary *CCBinary, module CCModuleType,
1128 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001129
Colin Cross97ba0732015-03-23 17:50:24 -07001130 return newCCDynamic(&binary.ccDynamic, module, hod, common.MultilibFirst,
1131 &binary.BinaryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001132}
1133
Colin Cross97ba0732015-03-23 17:50:24 -07001134func CCBinaryFactory() (blueprint.Module, []interface{}) {
1135 module := &CCBinary{}
1136
1137 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001138}
1139
Colin Cross21b9a242015-03-24 14:15:58 -07001140func (c *CCBinary) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
1141 flags = c.ccDynamic.Flags(ctx, flags)
1142
Colin Cross97ba0732015-03-23 17:50:24 -07001143 flags.CFlags = append(flags.CFlags, "-fpie")
1144
Colin Crossf6566ed2015-03-24 11:13:38 -07001145 if ctx.Device() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001146 linker := "/system/bin/linker"
Colin Cross97ba0732015-03-23 17:50:24 -07001147 if flags.Toolchain.Is64Bit() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001148 linker = "/system/bin/linker64"
1149 }
1150
Colin Cross97ba0732015-03-23 17:50:24 -07001151 flags.LdFlags = append(flags.LdFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -08001152 "-nostdlib",
1153 "-Bdynamic",
1154 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1155 "-Wl,--gc-sections",
1156 "-Wl,-z,nocopyreloc",
Colin Cross97ba0732015-03-23 17:50:24 -07001157 )
Colin Cross3f40fa42015-01-30 17:27:36 -08001158 }
1159
Colin Cross97ba0732015-03-23 17:50:24 -07001160 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001161}
1162
Colin Cross97ba0732015-03-23 17:50:24 -07001163func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1164 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001165
Colin Cross97ba0732015-03-23 17:50:24 -07001166 if !c.BinaryProperties.Static_executable && inList("libc", c.properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001167 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1168 "from static libs or set static_executable: true")
1169 }
1170
1171 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001172 c.out = outputFile
Colin Cross3f40fa42015-01-30 17:27:36 -08001173
Colin Cross97ba0732015-03-23 17:50:24 -07001174 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
1175 deps.LateStaticLibs, deps.WholeStaticLibs, deps.CrtBegin, deps.CrtEnd,
Colin Cross77b00fa2015-03-16 16:15:49 -07001176 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001177}
Colin Cross3f40fa42015-01-30 17:27:36 -08001178
Colin Cross97ba0732015-03-23 17:50:24 -07001179func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001180 ctx.InstallFile("bin", c.out)
1181}
1182
1183type ccTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001184 CCBinary
Colin Cross6b290692015-03-19 14:05:33 -07001185
1186 testProperties struct {
1187 // test_per_src: Create a separate test for each source file. Useful when there is
1188 // global state that can not be torn down and reset between each test suite.
1189 Test_per_src bool
1190 }
Dan Albertc403f7c2015-03-18 14:01:18 -07001191}
1192
Colin Cross21b9a242015-03-24 14:15:58 -07001193func (c *ccTest) Flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
1194 flags = c.CCBinary.Flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001195
Colin Cross97ba0732015-03-23 17:50:24 -07001196 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001197 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001198 flags.CFlags = append(flags.CFlags, "-O0", "-g")
1199 flags.LdLibs = append(flags.LdLibs, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001200 }
1201
1202 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross97ba0732015-03-23 17:50:24 -07001203 flags.IncludeDirs = append(flags.IncludeDirs,
Tim Kilbourn5ccc7302015-03-19 10:02:21 -07001204 filepath.Join(ctx.Config().(Config).SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001205
Colin Cross21b9a242015-03-24 14:15:58 -07001206 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001207}
1208
Colin Cross21b9a242015-03-24 14:15:58 -07001209func (c *ccTest) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1210 depNames = c.CCBinary.DepNames(ctx, depNames)
1211 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest", "libgtest_main")
1212 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001213}
1214
Colin Cross97ba0732015-03-23 17:50:24 -07001215func (c *ccTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001216 if ctx.Device() {
Tim Kilbourn5ccc7302015-03-19 10:02:21 -07001217 ctx.InstallFile("../data/nativetest/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001218 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001219 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001220 }
1221}
1222
Colin Cross97ba0732015-03-23 17:50:24 -07001223func CCTestFactory() (blueprint.Module, []interface{}) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001224 module := &ccTest{}
1225 return newCCDynamic(&module.ccDynamic, module, common.HostAndDeviceSupported,
Colin Cross97ba0732015-03-23 17:50:24 -07001226 common.MultilibFirst, &module.BinaryProperties, &module.testProperties)
Colin Cross6b290692015-03-19 14:05:33 -07001227}
1228
1229func TestPerSrcMutator(mctx blueprint.EarlyMutatorContext) {
1230 if test, ok := mctx.Module().(*ccTest); ok {
1231 if test.testProperties.Test_per_src {
1232 testNames := make([]string, len(test.properties.Srcs))
1233 for i, src := range test.properties.Srcs {
1234 testNames[i] = strings.TrimSuffix(src, filepath.Ext(src))
1235 }
1236 tests := mctx.CreateLocalVariations(testNames...)
1237 for i, src := range test.properties.Srcs {
1238 tests[i].(*ccTest).properties.Srcs = []string{src}
Colin Cross97ba0732015-03-23 17:50:24 -07001239 tests[i].(*ccTest).BinaryProperties.Stem = testNames[i]
Colin Cross6b290692015-03-19 14:05:33 -07001240 }
1241 }
1242 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001243}
1244
1245//
1246// Static library
1247//
1248
Colin Cross97ba0732015-03-23 17:50:24 -07001249func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1250 module := &CCLibrary{}
1251 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001252
Colin Cross97ba0732015-03-23 17:50:24 -07001253 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001254}
1255
1256//
1257// Shared libraries
1258//
1259
Colin Cross97ba0732015-03-23 17:50:24 -07001260func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1261 module := &CCLibrary{}
1262 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001263
Colin Cross97ba0732015-03-23 17:50:24 -07001264 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001265}
1266
1267//
1268// Host static library
1269//
1270
Colin Cross97ba0732015-03-23 17:50:24 -07001271func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1272 module := &CCLibrary{}
1273 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001274
Colin Cross97ba0732015-03-23 17:50:24 -07001275 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001276}
1277
1278//
1279// Host Shared libraries
1280//
1281
Colin Cross97ba0732015-03-23 17:50:24 -07001282func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1283 module := &CCLibrary{}
1284 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001285
Colin Cross97ba0732015-03-23 17:50:24 -07001286 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001287}
1288
1289//
1290// Host Binaries
1291//
1292
Colin Cross97ba0732015-03-23 17:50:24 -07001293func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1294 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001295
Colin Cross97ba0732015-03-23 17:50:24 -07001296 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001297}
1298
1299//
1300// Device libraries shipped with gcc
1301//
1302
1303type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001304 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001305}
1306
1307func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1308 // toolchain libraries can't have any dependencies
1309 return nil
1310}
1311
Colin Cross21b9a242015-03-24 14:15:58 -07001312func (*toolchainLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001313 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001314 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001315}
1316
Colin Cross97ba0732015-03-23 17:50:24 -07001317func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001318 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001319
Colin Cross97ba0732015-03-23 17:50:24 -07001320 module.LibraryProperties.BuildStatic = true
1321
Colin Cross21b9a242015-03-24 14:15:58 -07001322 return newCCBase(&module.ccBase, module, common.DeviceSupported, common.MultilibBoth,
1323 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001324}
1325
1326func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001327 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001328
1329 libName := ctx.ModuleName() + staticLibraryExtension
1330 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1331
1332 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1333
1334 c.out = outputFile
1335
1336 ctx.CheckbuildFile(outputFile)
1337}
1338
Colin Cross97ba0732015-03-23 17:50:24 -07001339func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001340 // Toolchain libraries do not get installed.
1341}
1342
Dan Albertbe961682015-03-18 23:38:50 -07001343// NDK prebuilt libraries.
1344//
1345// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1346// either (with the exception of the shared STLs, which are installed to the app's directory rather
1347// than to the system image).
1348
1349func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1350 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
1351 ctx.Config().(Config).SrcDir(), version, toolchain.Name())
1352}
1353
1354type ndkPrebuiltLibrary struct {
1355 CCLibrary
1356}
1357
1358func (*ndkPrebuiltLibrary) AndroidDynamicDependencies(
1359 ctx common.AndroidDynamicDependerModuleContext) []string {
1360
1361 // NDK libraries can't have any dependencies
1362 return nil
1363}
1364
1365func (*ndkPrebuiltLibrary) DepNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1366 // NDK libraries can't have any dependencies
1367 return CCDeps{}
1368}
1369
1370func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1371 module := &ndkPrebuiltLibrary{}
1372 module.LibraryProperties.BuildShared = true
1373 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1374}
1375
1376func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1377 deps CCDeps, objFiles []string) {
1378 // A null build step, but it sets up the output path.
1379 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1380 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1381 }
1382
1383 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1384 common.ModuleSrcDir(ctx))
1385
1386 // NDK prebuilt libraries are named like: ndk_LIBNAME.SDK_VERSION.
1387 // We want to translate to just LIBNAME.
1388 libName := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1389 libDir := getNdkLibDir(ctx, flags.Toolchain, c.properties.Sdk_version)
1390 c.out = filepath.Join(libDir, libName + sharedLibraryExtension)
1391}
1392
1393func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1394 // Toolchain libraries do not get installed.
1395}
1396
1397// The NDK STLs are slightly different from the prebuilt system libraries:
1398// * Are not specific to each platform version.
1399// * The libraries are not in a predictable location for each STL.
1400
1401type ndkPrebuiltStl struct {
1402 ndkPrebuiltLibrary
1403}
1404
1405type ndkPrebuiltStaticStl struct {
1406 ndkPrebuiltStl
1407}
1408
1409type ndkPrebuiltSharedStl struct {
1410 ndkPrebuiltStl
1411}
1412
1413func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1414 module := &ndkPrebuiltSharedStl{}
1415 module.LibraryProperties.BuildShared = true
1416 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1417}
1418
1419func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1420 module := &ndkPrebuiltStaticStl{}
1421 module.LibraryProperties.BuildStatic = true
1422 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1423}
1424
1425func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1426 gccVersion := toolchain.GccVersion()
1427 var libDir string
1428 switch stl {
1429 case "libstlport":
1430 libDir = "cxx-stl/stlport/libs"
1431 case "libc++":
1432 libDir = "cxx-stl/llvm-libc++/libs"
1433 case "libgnustl":
1434 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1435 }
1436
1437 if libDir != "" {
1438 ndkSrcRoot := ctx.Config().(Config).SrcDir() + "/prebuilts/ndk/current/sources"
1439 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1440 }
1441
1442 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1443 return ""
1444}
1445
1446func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1447 deps CCDeps, objFiles []string) {
1448 // A null build step, but it sets up the output path.
1449 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1450 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1451 }
1452
1453 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
1454 common.ModuleSrcDir(ctx))
1455
1456 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1457 libExt := sharedLibraryExtension
1458 if c.LibraryProperties.BuildStatic {
1459 libExt = staticLibraryExtension
1460 }
1461
1462 stlName := strings.TrimSuffix(libName, "_shared")
1463 stlName = strings.TrimSuffix(stlName, "_static")
1464 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1465 c.out = libDir + "/" + libName + libExt
1466}
1467
Colin Cross3f40fa42015-01-30 17:27:36 -08001468func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
Colin Cross97ba0732015-03-23 17:50:24 -07001469 if c, ok := mctx.Module().(ccLibraryInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001470 var modules []blueprint.Module
Colin Cross97ba0732015-03-23 17:50:24 -07001471 if c.ccLibrary().LibraryProperties.BuildStatic && c.ccLibrary().LibraryProperties.BuildShared {
Colin Cross3f40fa42015-01-30 17:27:36 -08001472 modules = mctx.CreateLocalVariations("static", "shared")
Colin Cross97ba0732015-03-23 17:50:24 -07001473 modules[0].(ccLibraryInterface).ccLibrary().LibraryProperties.IsStatic = true
1474 modules[1].(ccLibraryInterface).ccLibrary().LibraryProperties.IsShared = true
1475 } else if c.ccLibrary().LibraryProperties.BuildStatic {
Colin Cross3f40fa42015-01-30 17:27:36 -08001476 modules = mctx.CreateLocalVariations("static")
Colin Cross97ba0732015-03-23 17:50:24 -07001477 modules[0].(ccLibraryInterface).ccLibrary().LibraryProperties.IsStatic = true
1478 } else if c.ccLibrary().LibraryProperties.BuildShared {
Colin Cross3f40fa42015-01-30 17:27:36 -08001479 modules = mctx.CreateLocalVariations("shared")
Colin Cross97ba0732015-03-23 17:50:24 -07001480 modules[0].(ccLibraryInterface).ccLibrary().LibraryProperties.IsShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001481 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001482 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001483 }
Colin Cross97ba0732015-03-23 17:50:24 -07001484 primary := modules[0].(ccLibraryInterface).ccLibrary()
Colin Cross3f40fa42015-01-30 17:27:36 -08001485 for _, m := range modules {
Colin Cross97ba0732015-03-23 17:50:24 -07001486 m.(ccLibraryInterface).ccLibrary().primary = primary
1487 if m.(ccLibraryInterface).ccLibrary() != primary {
1488 m.(ccLibraryInterface).ccLibrary().properties.SkipCompileObjs = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001489 }
1490 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001491 }
1492}