blob: f64107b33cf293f0f27cdb2a9d64c770b8974980 [file] [log] [blame]
Colin Cross3f40fa42015-01-30 17:27:36 -08001//
2// Licensed under the Apache License, Version 2.0 (the "License");
3// you may not use this file except in compliance with the License.
4// You may obtain a copy of the License at
5//
6// http://www.apache.org/licenses/LICENSE-2.0
7//
8// Unless required by applicable law or agreed to in writing, software
9// distributed under the License is distributed on an "AS IS" BASIS,
10// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
11// See the License for the specific language governing permissions and
12// limitations under the License.
13
14package cc
15
16// This file contains the module types for compiling C/C++ for Android, and converts the properties
17// into the flags and filenames necessary to pass to the compiler. The final creation of the rules
18// is handled in builder.go
19
20import (
21 "blueprint"
22 "blueprint/pathtools"
23 "fmt"
24 "path/filepath"
25 "strings"
26
27 "android/soong/common"
28)
29
30type Config interface {
31 SrcDir() string
32 PrebuiltOS() string
33}
34
35var (
36 HostPrebuiltTag = pctx.VariableConfigMethod("HostPrebuiltTag", Config.PrebuiltOS)
37 SrcDir = pctx.VariableConfigMethod("SrcDir", Config.SrcDir)
38
39 LibcRoot = pctx.StaticVariable("LibcRoot", "${SrcDir}/bionic/libc")
40 LibmRoot = pctx.StaticVariable("LibmRoot", "${SrcDir}/bionic/libm")
41)
42
43// Flags used by lots of devices. Putting them in package static variables will save bytes in
44// build.ninja so they aren't repeated for every file
45var (
46 commonGlobalCflags = []string{
47 "-DANDROID",
48 "-fmessage-length=0",
49 "-W",
50 "-Wall",
51 "-Wno-unused",
52 "-Winit-self",
53 "-Wpointer-arith",
54
55 // COMMON_RELEASE_CFLAGS
56 "-DNDEBUG",
57 "-UDEBUG",
58 }
59
60 deviceGlobalCflags = []string{
61 // TARGET_ERROR_FLAGS
62 "-Werror=return-type",
63 "-Werror=non-virtual-dtor",
64 "-Werror=address",
65 "-Werror=sequence-point",
66 }
67
68 hostGlobalCflags = []string{}
69
70 commonGlobalCppflags = []string{
71 "-Wsign-promo",
72 "-std=gnu++11",
73 }
74)
75
76func init() {
77 pctx.StaticVariable("commonGlobalCflags", strings.Join(commonGlobalCflags, " "))
78 pctx.StaticVariable("deviceGlobalCflags", strings.Join(deviceGlobalCflags, " "))
79 pctx.StaticVariable("hostGlobalCflags", strings.Join(hostGlobalCflags, " "))
80
81 pctx.StaticVariable("commonGlobalCppflags", strings.Join(commonGlobalCppflags, " "))
82
83 pctx.StaticVariable("commonClangGlobalCflags",
84 strings.Join(clangFilterUnknownCflags(commonGlobalCflags), " "))
85 pctx.StaticVariable("deviceClangGlobalCflags",
86 strings.Join(clangFilterUnknownCflags(deviceGlobalCflags), " "))
87 pctx.StaticVariable("hostClangGlobalCflags",
88 strings.Join(clangFilterUnknownCflags(hostGlobalCflags), " "))
Tim Kilbournf2948142015-03-11 12:03:03 -070089 pctx.StaticVariable("commonClangGlobalCppflags",
90 strings.Join(clangFilterUnknownCflags(commonGlobalCppflags), " "))
Colin Cross3f40fa42015-01-30 17:27:36 -080091
92 // Everything in this list is a crime against abstraction and dependency tracking.
93 // Do not add anything to this list.
94 pctx.StaticVariable("commonGlobalIncludes", strings.Join([]string{
95 "-isystem ${SrcDir}/system/core/include",
96 "-isystem ${SrcDir}/hardware/libhardware/include",
97 "-isystem ${SrcDir}/hardware/libhardware_legacy/include",
98 "-isystem ${SrcDir}/hardware/ril/include",
99 "-isystem ${SrcDir}/libnativehelper/include",
100 "-isystem ${SrcDir}/frameworks/native/include",
101 "-isystem ${SrcDir}/frameworks/native/opengl/include",
102 "-isystem ${SrcDir}/frameworks/av/include",
103 "-isystem ${SrcDir}/frameworks/base/include",
104 }, " "))
105
106 pctx.StaticVariable("clangPath", "${SrcDir}/prebuilts/clang/${HostPrebuiltTag}/host/3.6/bin/")
107}
108
109// CcProperties describes properties used to compile all C or C++ modules
110type ccProperties struct {
111 // srcs: list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
112 Srcs []string `android:"arch_variant,arch_subtract"`
113
114 // cflags: list of module-specific flags that will be used for C and C++ compiles.
115 Cflags []string `android:"arch_variant"`
116
117 // cppflags: list of module-specific flags that will be used for C++ compiles
118 Cppflags []string `android:"arch_variant"`
119
120 // conlyflags: list of module-specific flags that will be used for C compiles
121 Conlyflags []string `android:"arch_variant"`
122
123 // asflags: list of module-specific flags that will be used for .S compiles
124 Asflags []string `android:"arch_variant"`
125
126 // ldflags: list of module-specific flags that will be used for all link steps
127 Ldflags []string `android:"arch_variant"`
128
129 // include_dirs: list of directories relative to the root of the source tree that will
130 // be added to the include path using -I.
131 // If possible, don't use this. If adding paths from the current directory use
132 // local_include_dirs, if adding paths from other modules use export_include_dirs in
133 // that module.
134 Include_dirs []string `android:"arch_variant"`
135
136 // local_include_dirs: list of directories relative to the Blueprints file that will
137 // be added to the include path using -I
138 Local_include_dirs []string `android:"arch_variant"`
139
140 // export_include_dirs: list of directories relative to the Blueprints file that will
141 // be added to the include path using -I for any module that links against this module
142 Export_include_dirs []string
143
144 // clang_cflags: list of module-specific flags that will be used for C and C++ compiles when
145 // compiling with clang
146 Clang_cflags []string `android:"arch_variant"`
147
148 // clang_asflags: list of module-specific flags that will be used for .S compiles when
149 // compiling with clang
150 Clang_asflags []string `android:"arch_variant"`
151
152 // system_shared_libs: list of system libraries that will be dynamically linked to
153 // shared library and executable modules. If unset, generally defaults to libc
154 // and libm. Set to [] to prevent linking against libc and libm.
155 System_shared_libs []string
156
157 // whole_static_libs: list of modules whose object files should be linked into this module
158 // in their entirety. For static library modules, all of the .o files from the intermediate
159 // directory of the dependency will be linked into this modules .a file. For a shared library,
160 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
161 Whole_static_libs []string `android:"arch_variant"`
162
163 // static_libs: list of modules that should be statically linked into this module.
164 Static_libs []string `android:"arch_variant"`
165
166 // shared_libs: list of modules that should be dynamically linked into this module.
167 Shared_libs []string `android:"arch_variant"`
168
169 // allow_undefined_symbols: allow the module to contain undefined symbols. By default,
170 // modules cannot contain undefined symbols that are not satisified by their immediate
171 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
172 // This flag should only be necessary for compiling low-level libraries like libc.
173 Allow_undefined_symbols bool
174
175 // nocrt: don't link in crt_begin and crt_end. This flag should only be necessary for
176 // compiling crt or libc.
177 Nocrt bool `android:"arch_variant"`
178
179 // no_default_compiler_flags: don't insert default compiler flags into asflags, cflags,
180 // cppflags, conlyflags, ldflags, or include_dirs
181 No_default_compiler_flags bool
182
183 // clang: compile module with clang instead of gcc
184 Clang bool `android:"arch_variant"`
185
186 // rtti: pass -frtti instead of -fno-rtti
187 Rtti bool
188
189 // host_ldlibs: -l arguments to pass to linker for host-provided shared libraries
190 Host_ldlibs []string `android:"arch_variant"`
191
192 // stl: select the STL library to use. Possible values are "libc++", "libc++_static",
193 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
194 // default
195 Stl string
196
197 // Set for combined shared/static libraries to prevent compiling object files a second time
198 SkipCompileObjs bool `blueprint:"mutated"`
199}
200
201type unusedProperties struct {
202 Asan bool
203 Native_coverage bool
204 Strip string
205 Tags []string
206 Required []string
207}
208
209// Building C/C++ code is handled by objects that satisfy this interface via composition
210type ccModuleType interface {
211 common.AndroidModule
212
213 // Return the cflags that are specific to this _type_ of module
214 moduleTypeCflags(common.AndroidModuleContext, toolchain) []string
215
216 // Return the ldflags that are specific to this _type_ of module
217 moduleTypeLdflags(common.AndroidModuleContext, toolchain) []string
218
219 // Create a ccDeps struct that collects the module dependency info. Can also
220 // modify ccFlags in order to add dependency include directories, etc.
221 collectDeps(common.AndroidModuleContext, ccFlags) (ccDeps, ccFlags)
222
223 // Compile objects into final module
224 compileModule(common.AndroidModuleContext, ccFlags, ccDeps, []string)
225
226 // Return the output file (.o, .a or .so) for use by other modules
227 outputFile() string
228}
229
230func (c *ccBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
231 toolchain := c.findToolchain(ctx)
232 if ctx.Failed() {
233 return
234 }
235
236 flags := c.flags(ctx, toolchain)
237 if ctx.Failed() {
238 return
239 }
240
241 flags = c.addStlFlags(ctx, flags)
242 if ctx.Failed() {
243 return
244 }
245
246 deps, flags := c.ccModuleType().collectDeps(ctx, flags)
247 if ctx.Failed() {
248 return
249 }
250
251 objFiles := c.compileObjs(ctx, flags, deps)
252 if ctx.Failed() {
253 return
254 }
255
256 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
257 if ctx.Failed() {
258 return
259 }
260}
261
262func (c *ccBase) ccModuleType() ccModuleType {
263 return c.module
264}
265
266var _ common.AndroidDynamicDepender = (*ccBase)(nil)
267
268func (c *ccBase) findToolchain(ctx common.AndroidModuleContext) toolchain {
269 arch := ctx.Arch()
270 factory := toolchainFactories[arch.HostOrDevice][arch.ArchType]
271 if factory == nil {
272 panic(fmt.Sprintf("Toolchain not found for %s arch %q",
273 arch.HostOrDevice.String(), arch.String()))
274 }
275 return factory(arch.ArchVariant, arch.CpuVariant)
276}
277
278type ccDeps struct {
Colin Cross77b00fa2015-03-16 16:15:49 -0700279 staticLibs, sharedLibs, lateStaticLibs, wholeStaticLibs, objFiles, includeDirs []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800280
281 crtBegin, crtEnd string
282}
283
284type ccFlags struct {
285 globalFlags []string
286 asFlags []string
287 cFlags []string
288 conlyFlags []string
289 cppFlags []string
290 ldFlags []string
291 ldLibs []string
292 includeDirs []string
293 nocrt bool
294 toolchain toolchain
295 clang bool
296
297 extraStaticLibs []string
298 extraSharedLibs []string
299}
300
301// ccBase contains the properties and members used by all C/C++ module types
302type ccBase struct {
303 common.AndroidModuleBase
304 module ccModuleType
305
306 properties ccProperties
307 unused unusedProperties
308
309 installPath string
310}
311
312func (c *ccBase) moduleTypeCflags(ctx common.AndroidModuleContext, toolchain toolchain) []string {
313 return nil
314}
315
316func (c *ccBase) moduleTypeLdflags(ctx common.AndroidModuleContext, toolchain toolchain) []string {
317 return nil
318}
319
320func (c *ccBase) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
321 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, c.properties.Whole_static_libs...)
322 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, c.properties.Static_libs...)
323 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, c.properties.Shared_libs...)
324
325 return nil
326}
327
328// Create a ccFlags struct that collects the compile flags from global values,
329// per-target values, module type values, and per-module Blueprints properties
330func (c *ccBase) flags(ctx common.AndroidModuleContext, toolchain toolchain) ccFlags {
331
332 arch := ctx.Arch()
333
334 flags := ccFlags{
335 cFlags: c.properties.Cflags,
336 cppFlags: c.properties.Cppflags,
337 conlyFlags: c.properties.Conlyflags,
338 ldFlags: c.properties.Ldflags,
339 asFlags: c.properties.Asflags,
340 nocrt: c.properties.Nocrt,
341 toolchain: toolchain,
342 clang: c.properties.Clang,
343 }
344
345 if arch.HostOrDevice.Host() {
346 // TODO: allow per-module clang disable for host
347 flags.clang = true
348 }
349
350 if flags.clang {
351 flags.cFlags = clangFilterUnknownCflags(flags.cFlags)
352 flags.cFlags = append(flags.cFlags, c.properties.Clang_cflags...)
353 flags.asFlags = append(flags.asFlags, c.properties.Clang_asflags...)
354 flags.cppFlags = clangFilterUnknownCflags(flags.cppFlags)
355 flags.conlyFlags = clangFilterUnknownCflags(flags.conlyFlags)
356 flags.ldFlags = clangFilterUnknownCflags(flags.ldFlags)
357
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700358 flags.cFlags = append(flags.cFlags, "${clangExtraCflags}")
359 flags.conlyFlags = append(flags.conlyFlags, "${clangExtraConlyflags}")
360 if arch.HostOrDevice.Device() {
361 flags.cFlags = append(flags.cFlags, "${clangExtraTargetCflags}")
362 }
363
Colin Cross3f40fa42015-01-30 17:27:36 -0800364 target := "-target " + toolchain.ClangTriple()
365 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
366
367 flags.cFlags = append(flags.cFlags, target, gccPrefix)
368 flags.asFlags = append(flags.asFlags, target, gccPrefix)
369 flags.ldFlags = append(flags.ldFlags, target, gccPrefix)
370
371 if arch.HostOrDevice.Host() {
372 gccToolchain := "--gcc-toolchain=" + toolchain.GccRoot()
373 sysroot := "--sysroot=" + filepath.Join(toolchain.GccRoot(), "sysroot")
374
375 // TODO: also need more -B, -L flags to make host builds hermetic
376 flags.cFlags = append(flags.cFlags, gccToolchain, sysroot)
377 flags.asFlags = append(flags.asFlags, gccToolchain, sysroot)
378 flags.ldFlags = append(flags.ldFlags, gccToolchain, sysroot)
379 }
380 }
381
382 flags.includeDirs = pathtools.PrefixPaths(c.properties.Include_dirs, ctx.Config().(Config).SrcDir())
383 localIncludeDirs := pathtools.PrefixPaths(c.properties.Local_include_dirs, common.ModuleSrcDir(ctx))
384 flags.includeDirs = append(flags.includeDirs, localIncludeDirs...)
385
386 if !c.properties.No_default_compiler_flags {
387 flags.includeDirs = append(flags.includeDirs, []string{
388 common.ModuleSrcDir(ctx),
389 common.ModuleOutDir(ctx),
390 common.ModuleGenDir(ctx),
391 }...)
392
393 if arch.HostOrDevice.Device() && !c.properties.Allow_undefined_symbols {
394 flags.ldFlags = append(flags.ldFlags, "-Wl,--no-undefined")
395 }
396
397 if flags.clang {
Tim Kilbournf2948142015-03-11 12:03:03 -0700398 flags.cppFlags = append(flags.cppFlags, "${commonClangGlobalCppflags}")
Colin Cross3f40fa42015-01-30 17:27:36 -0800399 flags.globalFlags = []string{
400 "${commonGlobalIncludes}",
401 toolchain.IncludeFlags(),
402 toolchain.ClangCflags(),
403 "${commonClangGlobalCflags}",
404 fmt.Sprintf("${%sClangGlobalCflags}", arch.HostOrDevice),
405 }
406 } else {
Tim Kilbournf2948142015-03-11 12:03:03 -0700407 flags.cppFlags = append(flags.cppFlags, "${commonGlobalCppflags}")
Colin Cross3f40fa42015-01-30 17:27:36 -0800408 flags.globalFlags = []string{
409 "${commonGlobalIncludes}",
410 toolchain.IncludeFlags(),
411 toolchain.Cflags(),
412 "${commonGlobalCflags}",
413 fmt.Sprintf("${%sGlobalCflags}", arch.HostOrDevice),
414 }
415 }
416
417 if arch.HostOrDevice.Host() {
418 flags.ldFlags = append(flags.ldFlags, c.properties.Host_ldlibs...)
419 }
420
421 if arch.HostOrDevice.Device() {
422 if c.properties.Rtti {
423 flags.cppFlags = append(flags.cppFlags, "-frtti")
424 } else {
425 flags.cppFlags = append(flags.cppFlags, "-fno-rtti")
426 }
427 }
428
429 flags.asFlags = append(flags.asFlags, "-D__ASSEMBLY__")
430
431 if flags.clang {
432 flags.cppFlags = append(flags.cppFlags, toolchain.ClangCppflags())
433 flags.ldFlags = append(flags.ldFlags, toolchain.ClangLdflags())
434 } else {
435 flags.cppFlags = append(flags.cppFlags, toolchain.Cppflags())
436 flags.ldFlags = append(flags.ldFlags, toolchain.Ldflags())
437 }
438 }
439
440 flags.cFlags = append(flags.cFlags, c.ccModuleType().moduleTypeCflags(ctx, toolchain)...)
441 flags.ldFlags = append(flags.ldFlags, c.ccModuleType().moduleTypeLdflags(ctx, toolchain)...)
442
443 // Optimization to reduce size of build.ninja
444 // Replace the long list of flags for each file with a module-local variable
445 ctx.Variable(pctx, "cflags", strings.Join(flags.cFlags, " "))
446 ctx.Variable(pctx, "cppflags", strings.Join(flags.cppFlags, " "))
447 ctx.Variable(pctx, "asflags", strings.Join(flags.asFlags, " "))
448 flags.cFlags = []string{"$cflags"}
449 flags.cppFlags = []string{"$cppflags"}
450 flags.asFlags = []string{"$asflags"}
451
452 return flags
453}
454
455// Modify ccFlags structs with STL library info
456func (c *ccBase) addStlFlags(ctx common.AndroidModuleContext, flags ccFlags) ccFlags {
457 if !c.properties.No_default_compiler_flags {
458 arch := ctx.Arch()
459 stl := "libc++" // TODO: mingw needs libstdc++
460 if c.properties.Stl != "" {
461 stl = c.properties.Stl
462 }
463
464 stlStatic := false
465 if strings.HasSuffix(stl, "_static") {
466 stlStatic = true
467 }
468
469 switch stl {
470 case "libc++", "libc++_static":
471 flags.cFlags = append(flags.cFlags, "-D_USING_LIBCXX")
472 flags.includeDirs = append(flags.includeDirs, "${SrcDir}/external/libcxx/include")
473 if arch.HostOrDevice.Host() {
474 flags.cppFlags = append(flags.cppFlags, "-nostdinc++")
475 flags.ldFlags = append(flags.ldFlags, "-nodefaultlibs")
476 flags.ldLibs = append(flags.ldLibs, "-lc", "-lm", "-lpthread")
477 }
478 if stlStatic {
479 flags.extraStaticLibs = append(flags.extraStaticLibs, "libc++_static")
480 } else {
481 flags.extraSharedLibs = append(flags.extraSharedLibs, "libc++")
482 }
483 case "stlport", "stlport_static":
484 if arch.HostOrDevice.Device() {
485 flags.includeDirs = append(flags.includeDirs,
486 "${SrcDir}/external/stlport/stlport",
487 "${SrcDir}/bionic/libstdc++/include",
488 "${SrcDir}/bionic")
489 if stlStatic {
490 flags.extraStaticLibs = append(flags.extraStaticLibs, "libstdc++", "libstlport_static")
491 } else {
492 flags.extraSharedLibs = append(flags.extraSharedLibs, "libstdc++", "libstlport")
493 }
494 }
495 case "ndk":
496 panic("TODO")
497 case "libstdc++":
498 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
499 // tree is in good enough shape to not need it.
500 // Host builds will use GNU libstdc++.
501 if arch.HostOrDevice.Device() {
502 flags.includeDirs = append(flags.includeDirs, "${SrcDir}/bionic/libstdc++/include")
503 flags.extraSharedLibs = append(flags.extraSharedLibs, "libstdc++")
504 }
505 case "none":
506 if arch.HostOrDevice.Host() {
507 flags.cppFlags = append(flags.cppFlags, "-nostdinc++")
508 flags.ldFlags = append(flags.ldFlags, "-nodefaultlibs")
509 flags.ldLibs = append(flags.ldLibs, "-lc", "-lm")
510 }
511 default:
512 ctx.ModuleErrorf("stl: %q is not a supported STL", stl)
513 }
514
515 }
516 return flags
517}
518
519// Compile a list of source files into objects a specified subdirectory
520func (c *ccBase) customCompileObjs(ctx common.AndroidModuleContext, flags ccFlags,
521 deps ccDeps, subdir string, srcFiles []string) []string {
522
523 srcFiles = pathtools.PrefixPaths(srcFiles, common.ModuleSrcDir(ctx))
524 srcFiles = common.ExpandGlobs(ctx, srcFiles)
525
526 return TransformSourceToObj(ctx, subdir, srcFiles, ccFlagsToBuilderFlags(flags))
527}
528
529// Compile files listed in c.properties.Srcs into objects
530func (c *ccBase) compileObjs(ctx common.AndroidModuleContext, flags ccFlags,
531 deps ccDeps) []string {
532
533 if c.properties.SkipCompileObjs {
534 return nil
535 }
536
537 return c.customCompileObjs(ctx, flags, deps, "", c.properties.Srcs)
538}
539
540func (c *ccBase) outputFile() string {
541 return ""
542}
543
544func (c *ccBase) collectDepsFromList(ctx common.AndroidModuleContext,
545 names []string) (modules []common.AndroidModule,
546 outputFiles []string, exportedIncludeDirs []string) {
547
548 for _, n := range names {
549 found := false
550 ctx.VisitDirectDeps(func(m blueprint.Module) {
551 otherName := ctx.OtherModuleName(m)
552 if otherName != n {
553 return
554 }
555
556 if a, ok := m.(ccModuleType); ok {
557 if a.Disabled() {
558 // If a cc_library host+device module depends on a library that exists as both
559 // cc_library_shared and cc_library_host_shared, it will end up with two
560 // dependencies with the same name, one of which is marked disabled for each
561 // of host and device. Ignore the disabled one.
562 return
563 }
564 if a.HostOrDevice() != ctx.Arch().HostOrDevice {
565 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
566 otherName)
567 return
568 }
569
570 if outputFile := a.outputFile(); outputFile != "" {
571 if found {
572 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
573 return
574 }
575 outputFiles = append(outputFiles, outputFile)
576 modules = append(modules, a)
577 if i, ok := a.(ccExportedIncludeDirsProducer); ok {
578 exportedIncludeDirs = append(exportedIncludeDirs, i.exportedIncludeDirs()...)
579 }
580 found = true
581 } else {
582 ctx.ModuleErrorf("module %q missing output file", otherName)
583 return
584 }
585 } else {
586 ctx.ModuleErrorf("module %q not an android module", otherName)
587 return
588 }
589 })
590 if !found {
591 ctx.ModuleErrorf("unsatisified dependency on %q", n)
592 }
593 }
594
595 return modules, outputFiles, exportedIncludeDirs
596}
597
598func (c *ccBase) collectDeps(ctx common.AndroidModuleContext, flags ccFlags) (ccDeps, ccFlags) {
599 var deps ccDeps
600 var newIncludeDirs []string
601
602 wholeStaticLibNames := c.properties.Whole_static_libs
603 _, deps.wholeStaticLibs, newIncludeDirs = c.collectDepsFromList(ctx, wholeStaticLibNames)
604
605 deps.includeDirs = append(deps.includeDirs, newIncludeDirs...)
606
607 staticLibNames := c.properties.Static_libs
608 staticLibNames = append(staticLibNames, flags.extraStaticLibs...)
609 _, deps.staticLibs, newIncludeDirs = c.collectDepsFromList(ctx, staticLibNames)
610 deps.includeDirs = append(deps.includeDirs, newIncludeDirs...)
611
612 return deps, flags
613}
614
615// ccDynamic contains the properties and members used by shared libraries and dynamic executables
616type ccDynamic struct {
617 ccBase
618}
619
620const defaultSystemSharedLibraries = "__default__"
621
622func (c *ccDynamic) systemSharedLibs() []string {
623
624 if len(c.properties.System_shared_libs) == 1 &&
625 c.properties.System_shared_libs[0] == defaultSystemSharedLibraries {
626
627 if c.HostOrDevice().Host() {
628 return []string{}
629 } else {
630 return []string{"libc", "libm"}
631 }
632 }
633 return c.properties.System_shared_libs
634}
635
636var (
637 stlSharedLibs = []string{"libc++", "libstlport", "libstdc++"}
638 stlSharedHostLibs = []string{"libc++"}
639 stlStaticLibs = []string{"libc++_static", "libstlport_static", "libstdc++"}
640 stlStaticHostLibs = []string{"libc++_static"}
641)
642
643func (c *ccDynamic) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
644 deps := c.ccBase.AndroidDynamicDependencies(ctx)
645
646 if c.HostOrDevice().Device() {
647 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, c.systemSharedLibs()...)
648 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}},
649 "libcompiler_rt-extras",
Colin Cross77b00fa2015-03-16 16:15:49 -0700650 "libgcov",
Colin Cross3f40fa42015-01-30 17:27:36 -0800651 "libatomic",
652 "libgcc")
653
654 if c.properties.Stl != "none" {
655 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, stlSharedLibs...)
656 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, stlStaticLibs...)
657 }
658 } else {
659 if c.properties.Stl != "none" {
660 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, stlSharedHostLibs...)
661 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, stlStaticHostLibs...)
662 }
663 }
664
665 return deps
666}
667
Colin Cross3f40fa42015-01-30 17:27:36 -0800668func (c *ccDynamic) collectDeps(ctx common.AndroidModuleContext, flags ccFlags) (ccDeps, ccFlags) {
669 var newIncludeDirs []string
670
671 deps, flags := c.ccBase.collectDeps(ctx, flags)
672
673 systemSharedLibs := c.systemSharedLibs()
674 sharedLibNames := make([]string, 0, len(c.properties.Shared_libs)+len(systemSharedLibs)+
675 len(flags.extraSharedLibs))
676 sharedLibNames = append(sharedLibNames, c.properties.Shared_libs...)
677 sharedLibNames = append(sharedLibNames, systemSharedLibs...)
678 sharedLibNames = append(sharedLibNames, flags.extraSharedLibs...)
679 _, deps.sharedLibs, newIncludeDirs = c.collectDepsFromList(ctx, sharedLibNames)
680 deps.includeDirs = append(deps.includeDirs, newIncludeDirs...)
681
682 if ctx.Arch().HostOrDevice.Device() {
683 var staticLibs []string
Colin Cross77b00fa2015-03-16 16:15:49 -0700684 staticLibNames := []string{"libcompiler_rt-extras"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800685 _, staticLibs, newIncludeDirs = c.collectDepsFromList(ctx, staticLibNames)
686 deps.staticLibs = append(deps.staticLibs, staticLibs...)
687 deps.includeDirs = append(deps.includeDirs, newIncludeDirs...)
Colin Cross77b00fa2015-03-16 16:15:49 -0700688
689 // libgcc and libatomic have to be last on the command line
690 staticLibNames = []string{"libgcov", "libatomic", "libgcc"}
691 _, staticLibs, newIncludeDirs = c.collectDepsFromList(ctx, staticLibNames)
692 deps.lateStaticLibs = append(deps.lateStaticLibs, staticLibs...)
693 deps.includeDirs = append(deps.includeDirs, newIncludeDirs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800694 }
695
696 ctx.VisitDirectDeps(func(m blueprint.Module) {
697 if obj, ok := m.(*ccObject); ok {
698 otherName := ctx.OtherModuleName(m)
699 if strings.HasPrefix(otherName, "crtbegin") {
700 if !c.properties.Nocrt {
701 deps.crtBegin = obj.outputFile()
702 }
703 } else if strings.HasPrefix(otherName, "crtend") {
704 if !c.properties.Nocrt {
705 deps.crtEnd = obj.outputFile()
706 }
707 } else {
708 ctx.ModuleErrorf("object module type only support for crtbegin and crtend, found %q",
709 ctx.OtherModuleName(m))
710 }
711 }
712 })
713
714 flags.includeDirs = append(flags.includeDirs, deps.includeDirs...)
715
716 return deps, flags
717}
718
719type ccExportedIncludeDirsProducer interface {
720 exportedIncludeDirs() []string
721}
722
723//
724// Combined static+shared libraries
725//
726
727type ccLibrary struct {
728 ccDynamic
729
730 primary *ccLibrary
731 primaryObjFiles []string
732 objFiles []string
733 exportIncludeDirs []string
734 out string
735
736 libraryProperties struct {
737 BuildStatic bool `blueprint:"mutated"`
738 BuildShared bool `blueprint:"mutated"`
739 IsShared bool `blueprint:"mutated"`
740 IsStatic bool `blueprint:"mutated"`
741
742 Static struct {
743 Srcs []string `android:"arch_variant"`
744 Cflags []string `android:"arch_variant"`
745 } `android:"arch_variant"`
746 Shared struct {
747 Srcs []string `android:"arch_variant"`
748 Cflags []string `android:"arch_variant"`
749 } `android:"arch_variant"`
750 }
751}
752
753func NewCCLibrary() (blueprint.Module, []interface{}) {
754 module := &ccLibrary{}
755 module.module = module
756 module.properties.System_shared_libs = []string{defaultSystemSharedLibraries}
757 module.libraryProperties.BuildShared = true
758 module.libraryProperties.BuildStatic = true
759
760 return common.InitAndroidModule(module, common.HostAndDeviceSupported, "both",
761 &module.properties, &module.unused, &module.libraryProperties)
762}
763
764func (c *ccLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
765 if c.libraryProperties.IsShared {
766 deps := c.ccDynamic.AndroidDynamicDependencies(ctx)
767 if c.HostOrDevice().Device() {
768 deps = append(deps, "crtbegin_so", "crtend_so")
769 }
770 return deps
771 } else {
772 return c.ccBase.AndroidDynamicDependencies(ctx)
773 }
774}
775
776func (c *ccLibrary) collectDeps(ctx common.AndroidModuleContext, flags ccFlags) (ccDeps, ccFlags) {
777 if c.libraryProperties.IsStatic {
778 deps, flags := c.ccBase.collectDeps(ctx, flags)
779 wholeStaticLibNames := c.properties.Whole_static_libs
780 wholeStaticLibs, _, _ := c.collectDepsFromList(ctx, wholeStaticLibNames)
781
782 for _, m := range wholeStaticLibs {
783 if staticLib, ok := m.(*ccLibrary); ok && staticLib.libraryProperties.IsStatic {
784 deps.objFiles = append(deps.objFiles, staticLib.allObjFiles()...)
785 } else {
786 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
787 }
788 }
789
790 return deps, flags
791 } else if c.libraryProperties.IsShared {
792 return c.ccDynamic.collectDeps(ctx, flags)
793 } else {
794 panic("Not shared or static")
795 }
796}
797
798func (c *ccLibrary) outputFile() string {
799 return c.out
800}
801
802func (c *ccLibrary) allObjFiles() []string {
803 return c.objFiles
804}
805
806func (c *ccLibrary) exportedIncludeDirs() []string {
807 return c.exportIncludeDirs
808}
809
810func (c *ccLibrary) moduleTypeCflags(ctx common.AndroidModuleContext, toolchain toolchain) []string {
811 return []string{"-fPIC"}
812}
813
814func (c *ccLibrary) moduleTypeLdflags(ctx common.AndroidModuleContext, toolchain toolchain) []string {
815 if c.libraryProperties.IsShared {
816 libName := ctx.ModuleName()
817 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
818 sharedFlag := "-Wl,-shared"
819 if c.properties.Clang || ctx.Arch().HostOrDevice.Host() {
820 sharedFlag = "-shared"
821 }
822 if ctx.Arch().HostOrDevice.Device() {
823 return []string{
824 "-nostdlib",
825 "-Wl,--gc-sections",
826 sharedFlag,
827 "-Wl,-soname," + libName,
828 }
829 } else {
830 return []string{
831 "-Wl,--gc-sections",
832 sharedFlag,
833 "-Wl,-soname," + libName,
834 }
835 }
836 } else {
837 return nil
838 }
839}
840
841func (c *ccLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
842 flags ccFlags, deps ccDeps, objFiles []string) {
843
844 staticFlags := flags
845 staticFlags.cFlags = append(staticFlags.cFlags, c.libraryProperties.Static.Cflags...)
846 objFilesStatic := c.customCompileObjs(ctx, staticFlags, deps, common.DeviceStaticLibrary,
847 c.libraryProperties.Static.Srcs)
848
849 objFiles = append(objFiles, objFilesStatic...)
850
851 var includeDirs []string
852
853 wholeStaticLibNames := c.properties.Whole_static_libs
854 wholeStaticLibs, _, newIncludeDirs := c.collectDepsFromList(ctx, wholeStaticLibNames)
855 includeDirs = append(includeDirs, newIncludeDirs...)
856
857 for _, m := range wholeStaticLibs {
858 if staticLib, ok := m.(*ccLibrary); ok && staticLib.libraryProperties.IsStatic {
859 objFiles = append(objFiles, staticLib.allObjFiles()...)
860 } else {
861 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
862 }
863 }
864
865 staticLibNames := c.properties.Static_libs
866 _, _, newIncludeDirs = c.collectDepsFromList(ctx, staticLibNames)
867 includeDirs = append(includeDirs, newIncludeDirs...)
868
869 ctx.VisitDirectDeps(func(m blueprint.Module) {
870 if obj, ok := m.(*ccObject); ok {
871 otherName := ctx.OtherModuleName(m)
872 if !strings.HasPrefix(otherName, "crtbegin") && !strings.HasPrefix(otherName, "crtend") {
873 objFiles = append(objFiles, obj.outputFile())
874 }
875 }
876 })
877
878 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
879
880 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
881
882 c.objFiles = objFiles
883 c.out = outputFile
884 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
885 common.ModuleSrcDir(ctx))
886
887 ctx.CheckbuildFile(outputFile)
888}
889
890func (c *ccLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
891 flags ccFlags, deps ccDeps, objFiles []string) {
892
893 sharedFlags := flags
894 sharedFlags.cFlags = append(sharedFlags.cFlags, c.libraryProperties.Shared.Cflags...)
895 objFilesShared := c.customCompileObjs(ctx, sharedFlags, deps, common.DeviceSharedLibrary,
896 c.libraryProperties.Shared.Srcs)
897
898 objFiles = append(objFiles, objFilesShared...)
899
900 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
901
Colin Cross77b00fa2015-03-16 16:15:49 -0700902 TransformObjToDynamicBinary(ctx, objFiles, deps.sharedLibs, deps.staticLibs,
903 deps.lateStaticLibs, deps.wholeStaticLibs, deps.crtBegin, deps.crtEnd,
904 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -0800905
906 c.out = outputFile
907 c.exportIncludeDirs = pathtools.PrefixPaths(c.properties.Export_include_dirs,
908 common.ModuleSrcDir(ctx))
909
910 installDir := "lib"
911 if flags.toolchain.Is64Bit() {
912 installDir = "lib64"
913 }
914
915 ctx.InstallFile(installDir, outputFile)
916}
917
918func (c *ccLibrary) compileModule(ctx common.AndroidModuleContext,
919 flags ccFlags, deps ccDeps, objFiles []string) {
920
921 // Reuse the object files from the matching static library if it exists
922 if c.primary == c {
923 c.primaryObjFiles = objFiles
924 } else {
925 objFiles = append([]string(nil), c.primary.primaryObjFiles...)
926 }
927
928 if c.libraryProperties.IsStatic {
929 c.compileStaticLibrary(ctx, flags, deps, objFiles)
930 } else {
931 c.compileSharedLibrary(ctx, flags, deps, objFiles)
932 }
933}
934
935//
936// Objects (for crt*.o)
937//
938
939type ccObject struct {
940 ccBase
941 out string
942}
943
944func NewCCObject() (blueprint.Module, []interface{}) {
945 module := &ccObject{}
946 module.module = module
947
948 return common.InitAndroidModule(module, common.DeviceSupported, "both",
949 &module.properties, &module.unused)
950}
951
952func (*ccObject) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
953 // object files can't have any dynamic dependencies
954 return nil
955}
956
957func (c *ccObject) collectDeps(ctx common.AndroidModuleContext, flags ccFlags) (ccDeps, ccFlags) {
958 deps, flags := c.ccBase.collectDeps(ctx, flags)
959 ctx.VisitDirectDeps(func(m blueprint.Module) {
960 if obj, ok := m.(*ccObject); ok {
961 deps.objFiles = append(deps.objFiles, obj.outputFile())
962 } else {
963 ctx.ModuleErrorf("Unknown module type for dependency %q", ctx.OtherModuleName(m))
964 }
965 })
966
967 return deps, flags
968}
969
970func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
971 flags ccFlags, deps ccDeps, objFiles []string) {
972
973 objFiles = append(objFiles, deps.objFiles...)
974
975 var outputFile string
976 if len(objFiles) == 1 {
977 outputFile = objFiles[0]
978 } else {
979 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+".o")
980 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
981 }
982
983 c.out = outputFile
984
985 ctx.CheckbuildFile(outputFile)
986}
987
988func (c *ccObject) outputFile() string {
989 return c.out
990}
991
992//
993// Executables
994//
995
996type ccBinary struct {
997 ccDynamic
998 binaryProperties binaryProperties
999}
1000
1001type binaryProperties struct {
1002 // static_executable: compile executable with -static
1003 Static_executable bool
1004
1005 // stem: set the name of the output
1006 Stem string `android:"arch_variant"`
1007
1008 // prefix_symbols: if set, add an extra objcopy --prefix-symbols= step
1009 Prefix_symbols string
1010}
1011
1012func (c *ccBinary) getStem(ctx common.AndroidModuleContext) string {
1013 if c.binaryProperties.Stem != "" {
1014 return c.binaryProperties.Stem
1015 }
1016 return ctx.ModuleName()
1017}
1018
1019func (c *ccBinary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1020 deps := c.ccDynamic.AndroidDynamicDependencies(ctx)
1021 if c.HostOrDevice().Device() {
1022 if c.binaryProperties.Static_executable {
1023 deps = append(deps, "crtbegin_static", "crtend_android")
1024 } else {
1025 deps = append(deps, "crtbegin_dynamic", "crtend_android")
1026 }
1027 }
1028 return deps
1029}
1030
1031func NewCCBinary() (blueprint.Module, []interface{}) {
1032 module := &ccBinary{}
1033 module.module = module
1034
1035 module.properties.System_shared_libs = []string{defaultSystemSharedLibraries}
1036 return common.InitAndroidModule(module, common.HostAndDeviceSupported, "first", &module.properties,
1037 &module.unused, &module.binaryProperties)
1038}
1039
1040func (c *ccBinary) moduleTypeCflags(ctx common.AndroidModuleContext, toolchain toolchain) []string {
1041 return []string{"-fpie"}
1042}
1043
1044func (c *ccBinary) moduleTypeLdflags(ctx common.AndroidModuleContext, toolchain toolchain) []string {
1045 if ctx.Arch().HostOrDevice.Device() {
1046 linker := "/system/bin/linker"
1047 if toolchain.Is64Bit() {
1048 linker = "/system/bin/linker64"
1049 }
1050
1051 return []string{
1052 "-nostdlib",
1053 "-Bdynamic",
1054 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1055 "-Wl,--gc-sections",
1056 "-Wl,-z,nocopyreloc",
1057 }
1058 }
1059
1060 return nil
1061}
1062
1063func (c *ccBinary) compileModule(ctx common.AndroidModuleContext,
1064 flags ccFlags, deps ccDeps, objFiles []string) {
1065
1066 if !c.binaryProperties.Static_executable && inList("libc", c.properties.Static_libs) {
1067 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1068 "from static libs or set static_executable: true")
1069 }
1070
1071 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
1072
Colin Cross77b00fa2015-03-16 16:15:49 -07001073 TransformObjToDynamicBinary(ctx, objFiles, deps.sharedLibs, deps.staticLibs,
1074 deps.lateStaticLibs, deps.wholeStaticLibs, deps.crtBegin, deps.crtEnd,
1075 ccFlagsToBuilderFlags(flags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001076
1077 ctx.InstallFile("bin", outputFile)
1078}
1079
1080//
1081// Static library
1082//
1083
1084func NewCCLibraryStatic() (blueprint.Module, []interface{}) {
1085 module := &ccLibrary{}
1086 module.module = module
1087 module.libraryProperties.BuildStatic = true
1088
1089 return common.InitAndroidModule(module, common.HostAndDeviceSupported, "both",
1090 &module.properties, &module.unused)
1091}
1092
1093//
1094// Shared libraries
1095//
1096
1097func NewCCLibraryShared() (blueprint.Module, []interface{}) {
1098 module := &ccLibrary{}
1099 module.module = module
1100 module.properties.System_shared_libs = []string{defaultSystemSharedLibraries}
1101 module.libraryProperties.BuildShared = true
1102
1103 return common.InitAndroidModule(module, common.HostAndDeviceSupported, "both",
1104 &module.properties, &module.unused)
1105}
1106
1107//
1108// Host static library
1109//
1110
1111func NewCCLibraryHostStatic() (blueprint.Module, []interface{}) {
1112 module := &ccLibrary{}
1113 module.module = module
1114 module.libraryProperties.BuildStatic = true
1115
1116 return common.InitAndroidModule(module, common.HostSupported, "both",
1117 &module.properties, &module.unused)
1118}
1119
1120//
1121// Host Shared libraries
1122//
1123
1124func NewCCLibraryHostShared() (blueprint.Module, []interface{}) {
1125 module := &ccLibrary{}
1126 module.module = module
1127 module.libraryProperties.BuildShared = true
1128
1129 return common.InitAndroidModule(module, common.HostSupported, "both",
1130 &module.properties, &module.unused)
1131}
1132
1133//
1134// Host Binaries
1135//
1136
1137func NewCCBinaryHost() (blueprint.Module, []interface{}) {
1138 module := &ccBinary{}
1139 module.module = module
1140
1141 return common.InitAndroidModule(module, common.HostSupported, "first",
1142 &module.properties, &module.unused)
1143}
1144
1145//
1146// Device libraries shipped with gcc
1147//
1148
1149type toolchainLibrary struct {
1150 ccLibrary
1151}
1152
1153func (*toolchainLibrary) AndroidDynamicDependencies(ctx common.AndroidDynamicDependerModuleContext) []string {
1154 // toolchain libraries can't have any dependencies
1155 return nil
1156}
1157
1158func (*toolchainLibrary) collectDeps(ctx common.AndroidModuleContext, flags ccFlags) (ccDeps, ccFlags) {
1159 // toolchain libraries can't have any dependencies
1160 return ccDeps{}, flags
1161}
1162
1163func NewToolchainLibrary() (blueprint.Module, []interface{}) {
1164 module := &toolchainLibrary{}
1165 module.module = module
1166 module.libraryProperties.BuildStatic = true
1167
1168 return common.InitAndroidModule(module, common.DeviceSupported, "both")
1169}
1170
1171func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
1172 flags ccFlags, deps ccDeps, objFiles []string) {
1173
1174 libName := ctx.ModuleName() + staticLibraryExtension
1175 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1176
1177 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1178
1179 c.out = outputFile
1180
1181 ctx.CheckbuildFile(outputFile)
1182}
1183
1184func LinkageMutator(mctx blueprint.EarlyMutatorContext) {
1185 if c, ok := mctx.Module().(*ccLibrary); ok {
1186 var modules []blueprint.Module
1187 if c.libraryProperties.BuildStatic && c.libraryProperties.BuildShared {
1188 modules = mctx.CreateLocalVariations("static", "shared")
1189 modules[0].(*ccLibrary).libraryProperties.IsStatic = true
1190 modules[1].(*ccLibrary).libraryProperties.IsShared = true
1191 } else if c.libraryProperties.BuildStatic {
1192 modules = mctx.CreateLocalVariations("static")
1193 modules[0].(*ccLibrary).libraryProperties.IsStatic = true
1194 } else if c.libraryProperties.BuildShared {
1195 modules = mctx.CreateLocalVariations("shared")
1196 modules[0].(*ccLibrary).libraryProperties.IsShared = true
1197 } else {
1198 panic("ccLibrary not static or shared")
1199 }
1200 primary := modules[0].(*ccLibrary)
1201 for _, m := range modules {
1202 m.(*ccLibrary).primary = primary
1203 if m != primary {
1204 m.(*ccLibrary).properties.SkipCompileObjs = true
1205 }
1206 }
1207 } else if _, ok := mctx.Module().(*toolchainLibrary); ok {
1208 mctx.CreateLocalVariations("static")
1209 }
1210}