blob: 42b0f2e5abc1ad51dc0fdc4b8d0f5bb7e7a90d4d [file] [log] [blame]
Colin Cross5049f022015-03-18 13:28:46 -07001// Copyright 2015 Google Inc. All rights reserved.
Colin Cross3f40fa42015-01-30 17:27:36 -08002//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package cc
16
17// This file contains the module types for compiling C/C++ for Android, and converts the properties
18// into the flags and filenames necessary to pass to the compiler. The final creation of the rules
19// is handled in builder.go
20
21import (
Colin Cross3f40fa42015-01-30 17:27:36 -080022 "fmt"
23 "path/filepath"
Colin Cross0af4b842015-04-30 16:36:18 -070024 "runtime"
Colin Cross3f40fa42015-01-30 17:27:36 -080025 "strings"
26
Colin Cross97ba0732015-03-23 17:50:24 -070027 "github.com/google/blueprint"
28 "github.com/google/blueprint/pathtools"
Colin Cross06a931b2015-10-28 17:23:31 -070029 "github.com/google/blueprint/proptools"
Colin Cross97ba0732015-03-23 17:50:24 -070030
Colin Cross463a90e2015-06-17 14:20:06 -070031 "android/soong"
Colin Cross3f40fa42015-01-30 17:27:36 -080032 "android/soong/common"
Colin Cross5049f022015-03-18 13:28:46 -070033 "android/soong/genrule"
Colin Cross3f40fa42015-01-30 17:27:36 -080034)
35
Colin Cross463a90e2015-06-17 14:20:06 -070036func init() {
37 soong.RegisterModuleType("cc_library_static", CCLibraryStaticFactory)
38 soong.RegisterModuleType("cc_library_shared", CCLibrarySharedFactory)
39 soong.RegisterModuleType("cc_library", CCLibraryFactory)
40 soong.RegisterModuleType("cc_object", CCObjectFactory)
41 soong.RegisterModuleType("cc_binary", CCBinaryFactory)
42 soong.RegisterModuleType("cc_test", CCTestFactory)
43 soong.RegisterModuleType("cc_benchmark", CCBenchmarkFactory)
Colin Crosscfad1192015-11-02 16:43:11 -080044 soong.RegisterModuleType("cc_defaults", CCDefaultsFactory)
Colin Cross463a90e2015-06-17 14:20:06 -070045
46 soong.RegisterModuleType("toolchain_library", ToolchainLibraryFactory)
47 soong.RegisterModuleType("ndk_prebuilt_library", NdkPrebuiltLibraryFactory)
48 soong.RegisterModuleType("ndk_prebuilt_object", NdkPrebuiltObjectFactory)
49 soong.RegisterModuleType("ndk_prebuilt_static_stl", NdkPrebuiltStaticStlFactory)
50 soong.RegisterModuleType("ndk_prebuilt_shared_stl", NdkPrebuiltSharedStlFactory)
51
52 soong.RegisterModuleType("cc_library_host_static", CCLibraryHostStaticFactory)
53 soong.RegisterModuleType("cc_library_host_shared", CCLibraryHostSharedFactory)
54 soong.RegisterModuleType("cc_binary_host", CCBinaryHostFactory)
55 soong.RegisterModuleType("cc_test_host", CCTestHostFactory)
56 soong.RegisterModuleType("cc_benchmark_host", CCBenchmarkHostFactory)
57
58 // LinkageMutator must be registered after common.ArchMutator, but that is guaranteed by
59 // the Go initialization order because this package depends on common, so common's init
60 // functions will run first.
Colin Cross6362e272015-10-29 15:25:03 -070061 common.RegisterBottomUpMutator("link", linkageMutator)
62 common.RegisterBottomUpMutator("test_per_src", testPerSrcMutator)
63 common.RegisterBottomUpMutator("deps", depsMutator)
Colin Cross463a90e2015-06-17 14:20:06 -070064}
65
Colin Cross3f40fa42015-01-30 17:27:36 -080066var (
Colin Cross1332b002015-04-07 17:11:30 -070067 HostPrebuiltTag = pctx.VariableConfigMethod("HostPrebuiltTag", common.Config.PrebuiltOS)
68 SrcDir = pctx.VariableConfigMethod("SrcDir", common.Config.SrcDir)
Colin Cross3f40fa42015-01-30 17:27:36 -080069
Dan Willemsen87b17d12015-07-14 00:39:06 -070070 LibcRoot = pctx.StaticVariable("LibcRoot", "bionic/libc")
71 LibmRoot = pctx.StaticVariable("LibmRoot", "bionic/libm")
Colin Cross3f40fa42015-01-30 17:27:36 -080072)
73
74// Flags used by lots of devices. Putting them in package static variables will save bytes in
75// build.ninja so they aren't repeated for every file
76var (
77 commonGlobalCflags = []string{
78 "-DANDROID",
79 "-fmessage-length=0",
80 "-W",
81 "-Wall",
82 "-Wno-unused",
83 "-Winit-self",
84 "-Wpointer-arith",
Dan Willemsene6540452015-10-20 15:21:33 -070085 "-fdiagnostics-color",
86 "-fdebug-prefix-map=/proc/self/cwd=",
Colin Cross3f40fa42015-01-30 17:27:36 -080087
88 // COMMON_RELEASE_CFLAGS
89 "-DNDEBUG",
90 "-UDEBUG",
91 }
92
93 deviceGlobalCflags = []string{
94 // TARGET_ERROR_FLAGS
95 "-Werror=return-type",
96 "-Werror=non-virtual-dtor",
97 "-Werror=address",
98 "-Werror=sequence-point",
99 }
100
101 hostGlobalCflags = []string{}
102
103 commonGlobalCppflags = []string{
104 "-Wsign-promo",
Dan Willemsen3bf6b472015-09-11 17:41:10 -0700105 }
106
107 illegalFlags = []string{
108 "-w",
Colin Cross3f40fa42015-01-30 17:27:36 -0800109 }
110)
111
112func init() {
113 pctx.StaticVariable("commonGlobalCflags", strings.Join(commonGlobalCflags, " "))
114 pctx.StaticVariable("deviceGlobalCflags", strings.Join(deviceGlobalCflags, " "))
115 pctx.StaticVariable("hostGlobalCflags", strings.Join(hostGlobalCflags, " "))
116
117 pctx.StaticVariable("commonGlobalCppflags", strings.Join(commonGlobalCppflags, " "))
118
119 pctx.StaticVariable("commonClangGlobalCflags",
120 strings.Join(clangFilterUnknownCflags(commonGlobalCflags), " "))
121 pctx.StaticVariable("deviceClangGlobalCflags",
122 strings.Join(clangFilterUnknownCflags(deviceGlobalCflags), " "))
123 pctx.StaticVariable("hostClangGlobalCflags",
124 strings.Join(clangFilterUnknownCflags(hostGlobalCflags), " "))
Tim Kilbournf2948142015-03-11 12:03:03 -0700125 pctx.StaticVariable("commonClangGlobalCppflags",
126 strings.Join(clangFilterUnknownCflags(commonGlobalCppflags), " "))
Colin Cross3f40fa42015-01-30 17:27:36 -0800127
128 // Everything in this list is a crime against abstraction and dependency tracking.
129 // Do not add anything to this list.
130 pctx.StaticVariable("commonGlobalIncludes", strings.Join([]string{
131 "-isystem ${SrcDir}/system/core/include",
132 "-isystem ${SrcDir}/hardware/libhardware/include",
133 "-isystem ${SrcDir}/hardware/libhardware_legacy/include",
134 "-isystem ${SrcDir}/hardware/ril/include",
135 "-isystem ${SrcDir}/libnativehelper/include",
136 "-isystem ${SrcDir}/frameworks/native/include",
137 "-isystem ${SrcDir}/frameworks/native/opengl/include",
138 "-isystem ${SrcDir}/frameworks/av/include",
139 "-isystem ${SrcDir}/frameworks/base/include",
140 }, " "))
141
Dan Willemsenb30a8112015-11-24 13:02:49 -0800142 pctx.StaticVariable("clangPath", "${SrcDir}/prebuilts/clang/host/${HostPrebuiltTag}/3.8/bin/")
Colin Cross3f40fa42015-01-30 17:27:36 -0800143}
144
Colin Cross6362e272015-10-29 15:25:03 -0700145type CCModuleContext common.AndroidBaseContext
146
Colin Cross3f40fa42015-01-30 17:27:36 -0800147// Building C/C++ code is handled by objects that satisfy this interface via composition
Colin Cross97ba0732015-03-23 17:50:24 -0700148type CCModuleType interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800149 common.AndroidModule
150
Colin Crossfa138792015-04-24 17:31:52 -0700151 // Modify property values after parsing Blueprints file but before starting dependency
152 // resolution or build rule generation
Colin Cross6362e272015-10-29 15:25:03 -0700153 ModifyProperties(CCModuleContext)
Colin Crossfa138792015-04-24 17:31:52 -0700154
Colin Cross21b9a242015-03-24 14:15:58 -0700155 // Modify the ccFlags
Colin Cross0676e2d2015-04-24 17:39:18 -0700156 flags(common.AndroidModuleContext, CCFlags) CCFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800157
Colin Cross6362e272015-10-29 15:25:03 -0700158 // Return list of dependency names for use in depsMutator
Colin Cross0676e2d2015-04-24 17:39:18 -0700159 depNames(common.AndroidBaseContext, CCDeps) CCDeps
Colin Cross3f40fa42015-01-30 17:27:36 -0800160
Colin Cross6362e272015-10-29 15:25:03 -0700161 // Add dynamic dependencies
162 depsMutator(common.AndroidBottomUpMutatorContext)
163
Colin Cross3f40fa42015-01-30 17:27:36 -0800164 // Compile objects into final module
Colin Cross97ba0732015-03-23 17:50:24 -0700165 compileModule(common.AndroidModuleContext, CCFlags, CCDeps, []string)
Colin Cross3f40fa42015-01-30 17:27:36 -0800166
Dan Albertc403f7c2015-03-18 14:01:18 -0700167 // Install the built module.
Colin Cross97ba0732015-03-23 17:50:24 -0700168 installModule(common.AndroidModuleContext, CCFlags)
Dan Albertc403f7c2015-03-18 14:01:18 -0700169
Colin Cross3f40fa42015-01-30 17:27:36 -0800170 // Return the output file (.o, .a or .so) for use by other modules
171 outputFile() string
172}
173
Colin Cross97ba0732015-03-23 17:50:24 -0700174type CCDeps struct {
Colin Crossa48f71f2015-11-16 18:00:41 -0800175 StaticLibs, SharedLibs, LateStaticLibs, WholeStaticLibs, ObjFiles, Cflags, ReexportedCflags []string
Colin Crossc472d572015-03-17 15:06:21 -0700176
Colin Cross21b9a242015-03-24 14:15:58 -0700177 WholeStaticLibObjFiles []string
178
Colin Cross97ba0732015-03-23 17:50:24 -0700179 CrtBegin, CrtEnd string
Colin Crossc472d572015-03-17 15:06:21 -0700180}
181
Colin Cross97ba0732015-03-23 17:50:24 -0700182type CCFlags struct {
Colin Cross28344522015-04-22 13:07:53 -0700183 GlobalFlags []string // Flags that apply to C, C++, and assembly source files
184 AsFlags []string // Flags that apply to assembly source files
185 CFlags []string // Flags that apply to C and C++ source files
186 ConlyFlags []string // Flags that apply to C source files
187 CppFlags []string // Flags that apply to C++ source files
188 YaccFlags []string // Flags that apply to Yacc source files
189 LdFlags []string // Flags that apply to linker command lines
190
191 Nocrt bool
192 Toolchain Toolchain
193 Clang bool
Colin Crossc472d572015-03-17 15:06:21 -0700194}
195
Colin Cross7d5136f2015-05-11 13:39:40 -0700196// Properties used to compile all C or C++ modules
197type CCBaseProperties struct {
198 // list of source files used to compile the C/C++ module. May be .c, .cpp, or .S files.
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700199 Srcs []string `android:"arch_variant"`
200
201 // list of source files that should not be used to build the C/C++ module.
202 // This is most useful in the arch/multilib variants to remove non-common files
203 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700204
205 // list of module-specific flags that will be used for C and C++ compiles.
206 Cflags []string `android:"arch_variant"`
207
208 // list of module-specific flags that will be used for C++ compiles
209 Cppflags []string `android:"arch_variant"`
210
211 // list of module-specific flags that will be used for C compiles
212 Conlyflags []string `android:"arch_variant"`
213
214 // list of module-specific flags that will be used for .S compiles
215 Asflags []string `android:"arch_variant"`
216
217 // list of module-specific flags that will be used for .y and .yy compiles
218 Yaccflags []string
219
220 // list of module-specific flags that will be used for all link steps
221 Ldflags []string `android:"arch_variant"`
222
223 // the instruction set architecture to use to compile the C/C++
224 // module.
225 Instruction_set string `android:"arch_variant"`
226
227 // list of directories relative to the root of the source tree that will
228 // be added to the include path using -I.
229 // If possible, don't use this. If adding paths from the current directory use
230 // local_include_dirs, if adding paths from other modules use export_include_dirs in
231 // that module.
232 Include_dirs []string `android:"arch_variant"`
233
Colin Cross39d97f22015-09-14 12:30:50 -0700234 // list of files relative to the root of the source tree that will be included
235 // using -include.
236 // If possible, don't use this.
237 Include_files []string `android:"arch_variant"`
238
Colin Cross7d5136f2015-05-11 13:39:40 -0700239 // list of directories relative to the Blueprints file that will
240 // be added to the include path using -I
241 Local_include_dirs []string `android:"arch_variant"`
242
Colin Cross39d97f22015-09-14 12:30:50 -0700243 // list of files relative to the Blueprints file that will be included
244 // using -include.
245 // If possible, don't use this.
246 Local_include_files []string `android:"arch_variant"`
247
Colin Cross7d5136f2015-05-11 13:39:40 -0700248 // list of directories relative to the Blueprints file that will
249 // be added to the include path using -I for any module that links against this module
250 Export_include_dirs []string `android:"arch_variant"`
251
252 // list of module-specific flags that will be used for C and C++ compiles when
253 // compiling with clang
254 Clang_cflags []string `android:"arch_variant"`
255
256 // list of module-specific flags that will be used for .S compiles when
257 // compiling with clang
258 Clang_asflags []string `android:"arch_variant"`
259
260 // list of system libraries that will be dynamically linked to
261 // shared library and executable modules. If unset, generally defaults to libc
262 // and libm. Set to [] to prevent linking against libc and libm.
263 System_shared_libs []string
264
265 // list of modules whose object files should be linked into this module
266 // in their entirety. For static library modules, all of the .o files from the intermediate
267 // directory of the dependency will be linked into this modules .a file. For a shared library,
268 // the dependency's .a file will be linked into this module using -Wl,--whole-archive.
269 Whole_static_libs []string `android:"arch_variant"`
270
271 // list of modules that should be statically linked into this module.
272 Static_libs []string `android:"arch_variant"`
273
274 // list of modules that should be dynamically linked into this module.
275 Shared_libs []string `android:"arch_variant"`
276
277 // allow the module to contain undefined symbols. By default,
278 // modules cannot contain undefined symbols that are not satisified by their immediate
279 // dependencies. Set this flag to true to remove --no-undefined from the linker flags.
280 // This flag should only be necessary for compiling low-level libraries like libc.
Colin Cross06a931b2015-10-28 17:23:31 -0700281 Allow_undefined_symbols *bool
Colin Cross7d5136f2015-05-11 13:39:40 -0700282
283 // don't link in crt_begin and crt_end. This flag should only be necessary for
284 // compiling crt or libc.
Colin Cross06a931b2015-10-28 17:23:31 -0700285 Nocrt *bool `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700286
Dan Willemsend67be222015-09-16 15:19:33 -0700287 // don't link in libgcc.a
Colin Cross06a931b2015-10-28 17:23:31 -0700288 No_libgcc *bool
Dan Willemsend67be222015-09-16 15:19:33 -0700289
Colin Cross7d5136f2015-05-11 13:39:40 -0700290 // don't insert default compiler flags into asflags, cflags,
291 // cppflags, conlyflags, ldflags, or include_dirs
Colin Cross06a931b2015-10-28 17:23:31 -0700292 No_default_compiler_flags *bool
Colin Cross7d5136f2015-05-11 13:39:40 -0700293
294 // compile module with clang instead of gcc
Colin Cross06a931b2015-10-28 17:23:31 -0700295 Clang *bool `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700296
297 // pass -frtti instead of -fno-rtti
Colin Cross06a931b2015-10-28 17:23:31 -0700298 Rtti *bool
Colin Cross7d5136f2015-05-11 13:39:40 -0700299
300 // -l arguments to pass to linker for host-provided shared libraries
301 Host_ldlibs []string `android:"arch_variant"`
302
303 // select the STL library to use. Possible values are "libc++", "libc++_static",
304 // "stlport", "stlport_static", "ndk", "libstdc++", or "none". Leave blank to select the
305 // default
306 Stl string
307
308 // Set for combined shared/static libraries to prevent compiling object files a second time
309 SkipCompileObjs bool `blueprint:"mutated"`
310
311 Debug, Release struct {
312 // list of module-specific flags that will be used for C and C++ compiles in debug or
313 // release builds
314 Cflags []string `android:"arch_variant"`
315 } `android:"arch_variant"`
316
317 // Minimum sdk version supported when compiling against the ndk
318 Sdk_version string
319
320 // install to a subdirectory of the default install path for the module
321 Relative_install_path string
322}
323
Colin Crosscfad1192015-11-02 16:43:11 -0800324type CCUnusedProperties struct {
325 Native_coverage *bool
326 Required []string
327 Sanitize []string `android:"arch_variant"`
328 Sanitize_recover []string
329 Strip string
330 Tags []string
331}
332
Colin Crossfa138792015-04-24 17:31:52 -0700333// CCBase contains the properties and members used by all C/C++ module types, and implements
Colin Crossc472d572015-03-17 15:06:21 -0700334// the blueprint.Module interface. It expects to be embedded into an outer specialization struct,
335// and uses a ccModuleType interface to that struct to create the build steps.
Colin Crossfa138792015-04-24 17:31:52 -0700336type CCBase struct {
Colin Crossc472d572015-03-17 15:06:21 -0700337 common.AndroidModuleBase
Colin Crosscfad1192015-11-02 16:43:11 -0800338 common.DefaultableModule
Colin Cross97ba0732015-03-23 17:50:24 -0700339 module CCModuleType
Colin Crossc472d572015-03-17 15:06:21 -0700340
Colin Cross7d5136f2015-05-11 13:39:40 -0700341 Properties CCBaseProperties
Colin Crossfa138792015-04-24 17:31:52 -0700342
Colin Crosscfad1192015-11-02 16:43:11 -0800343 unused CCUnusedProperties
Colin Crossc472d572015-03-17 15:06:21 -0700344
345 installPath string
Colin Cross74d1ec02015-04-28 13:30:13 -0700346
347 savedDepNames CCDeps
Colin Crossc472d572015-03-17 15:06:21 -0700348}
349
Colin Crossfa138792015-04-24 17:31:52 -0700350func newCCBase(base *CCBase, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700351 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
352
353 base.module = module
354
Colin Crossfa138792015-04-24 17:31:52 -0700355 props = append(props, &base.Properties, &base.unused)
Colin Crossc472d572015-03-17 15:06:21 -0700356
Colin Crosscfad1192015-11-02 16:43:11 -0800357 _, props = common.InitAndroidArchModule(module, hod, multilib, props...)
358
359 return common.InitDefaultableModule(module, base, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700360}
361
Colin Crossfa138792015-04-24 17:31:52 -0700362func (c *CCBase) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800363 toolchain := c.findToolchain(ctx)
364 if ctx.Failed() {
365 return
366 }
367
Colin Cross21b9a242015-03-24 14:15:58 -0700368 flags := c.collectFlags(ctx, toolchain)
Colin Cross3f40fa42015-01-30 17:27:36 -0800369 if ctx.Failed() {
370 return
371 }
372
Colin Cross74d1ec02015-04-28 13:30:13 -0700373 deps := c.depsToPaths(ctx, c.savedDepNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800374 if ctx.Failed() {
375 return
376 }
377
Colin Cross28344522015-04-22 13:07:53 -0700378 flags.CFlags = append(flags.CFlags, deps.Cflags...)
Colin Crossed9f8682015-03-18 17:17:35 -0700379
Colin Cross581c1892015-04-07 16:50:10 -0700380 objFiles := c.compileObjs(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800381 if ctx.Failed() {
382 return
383 }
384
Colin Cross581c1892015-04-07 16:50:10 -0700385 generatedObjFiles := c.compileGeneratedObjs(ctx, flags)
Colin Cross5049f022015-03-18 13:28:46 -0700386 if ctx.Failed() {
387 return
388 }
389
390 objFiles = append(objFiles, generatedObjFiles...)
391
Colin Cross3f40fa42015-01-30 17:27:36 -0800392 c.ccModuleType().compileModule(ctx, flags, deps, objFiles)
393 if ctx.Failed() {
394 return
395 }
Dan Albertc403f7c2015-03-18 14:01:18 -0700396
397 c.ccModuleType().installModule(ctx, flags)
398 if ctx.Failed() {
399 return
400 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800401}
402
Colin Crossfa138792015-04-24 17:31:52 -0700403func (c *CCBase) ccModuleType() CCModuleType {
Colin Cross3f40fa42015-01-30 17:27:36 -0800404 return c.module
405}
406
Colin Crossfa138792015-04-24 17:31:52 -0700407func (c *CCBase) findToolchain(ctx common.AndroidModuleContext) Toolchain {
Colin Cross3f40fa42015-01-30 17:27:36 -0800408 arch := ctx.Arch()
Colin Crossd3ba0392015-05-07 14:11:29 -0700409 hod := ctx.HostOrDevice()
410 factory := toolchainFactories[hod][arch.ArchType]
Colin Cross3f40fa42015-01-30 17:27:36 -0800411 if factory == nil {
Colin Crosseeabb892015-11-20 13:07:51 -0800412 ctx.ModuleErrorf("Toolchain not found for %s arch %q", hod.String(), arch.String())
413 return nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800414 }
Colin Crossc5c24ad2015-11-20 15:35:00 -0800415 return factory(arch)
Colin Cross3f40fa42015-01-30 17:27:36 -0800416}
417
Colin Cross6362e272015-10-29 15:25:03 -0700418func (c *CCBase) ModifyProperties(ctx CCModuleContext) {
Colin Crossfa138792015-04-24 17:31:52 -0700419}
420
Colin Crosse11befc2015-04-27 17:49:17 -0700421func (c *CCBase) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crossfa138792015-04-24 17:31:52 -0700422 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.Properties.Whole_static_libs...)
423 depNames.StaticLibs = append(depNames.StaticLibs, c.Properties.Static_libs...)
424 depNames.SharedLibs = append(depNames.SharedLibs, c.Properties.Shared_libs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700425
Colin Cross21b9a242015-03-24 14:15:58 -0700426 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800427}
428
Colin Cross6362e272015-10-29 15:25:03 -0700429func (c *CCBase) depsMutator(ctx common.AndroidBottomUpMutatorContext) {
Colin Cross74d1ec02015-04-28 13:30:13 -0700430 c.savedDepNames = c.module.depNames(ctx, CCDeps{})
431 c.savedDepNames.WholeStaticLibs = lastUniqueElements(c.savedDepNames.WholeStaticLibs)
432 c.savedDepNames.StaticLibs = lastUniqueElements(c.savedDepNames.StaticLibs)
433 c.savedDepNames.SharedLibs = lastUniqueElements(c.savedDepNames.SharedLibs)
434
435 staticLibs := c.savedDepNames.WholeStaticLibs
436 staticLibs = append(staticLibs, c.savedDepNames.StaticLibs...)
437 staticLibs = append(staticLibs, c.savedDepNames.LateStaticLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700438 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, staticLibs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800439
Colin Cross74d1ec02015-04-28 13:30:13 -0700440 ctx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, c.savedDepNames.SharedLibs...)
Colin Cross21b9a242015-03-24 14:15:58 -0700441
Colin Cross6362e272015-10-29 15:25:03 -0700442 ctx.AddDependency(ctx.Module(), c.savedDepNames.ObjFiles...)
Colin Cross74d1ec02015-04-28 13:30:13 -0700443 if c.savedDepNames.CrtBegin != "" {
Colin Cross6362e272015-10-29 15:25:03 -0700444 ctx.AddDependency(ctx.Module(), c.savedDepNames.CrtBegin)
Colin Cross21b9a242015-03-24 14:15:58 -0700445 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700446 if c.savedDepNames.CrtEnd != "" {
Colin Cross6362e272015-10-29 15:25:03 -0700447 ctx.AddDependency(ctx.Module(), c.savedDepNames.CrtEnd)
Colin Cross21b9a242015-03-24 14:15:58 -0700448 }
Colin Cross6362e272015-10-29 15:25:03 -0700449}
Colin Cross21b9a242015-03-24 14:15:58 -0700450
Colin Cross6362e272015-10-29 15:25:03 -0700451func depsMutator(ctx common.AndroidBottomUpMutatorContext) {
452 if c, ok := ctx.Module().(CCModuleType); ok {
453 c.ModifyProperties(ctx)
454 c.depsMutator(ctx)
455 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800456}
457
458// Create a ccFlags struct that collects the compile flags from global values,
459// per-target values, module type values, and per-module Blueprints properties
Colin Crossfa138792015-04-24 17:31:52 -0700460func (c *CCBase) collectFlags(ctx common.AndroidModuleContext, toolchain Toolchain) CCFlags {
Colin Cross97ba0732015-03-23 17:50:24 -0700461 flags := CCFlags{
Colin Crossfa138792015-04-24 17:31:52 -0700462 CFlags: c.Properties.Cflags,
463 CppFlags: c.Properties.Cppflags,
464 ConlyFlags: c.Properties.Conlyflags,
465 LdFlags: c.Properties.Ldflags,
466 AsFlags: c.Properties.Asflags,
467 YaccFlags: c.Properties.Yaccflags,
Colin Cross06a931b2015-10-28 17:23:31 -0700468 Nocrt: Bool(c.Properties.Nocrt),
Colin Cross97ba0732015-03-23 17:50:24 -0700469 Toolchain: toolchain,
Colin Cross06a931b2015-10-28 17:23:31 -0700470 Clang: Bool(c.Properties.Clang),
Colin Cross3f40fa42015-01-30 17:27:36 -0800471 }
Colin Cross28344522015-04-22 13:07:53 -0700472
473 // Include dir cflags
Colin Crossf2298272015-05-12 11:36:53 -0700474 common.CheckSrcDirsExist(ctx, c.Properties.Include_dirs, "include_dirs")
475 common.CheckModuleSrcDirsExist(ctx, c.Properties.Local_include_dirs, "local_include_dirs")
476
Colin Crossfa138792015-04-24 17:31:52 -0700477 rootIncludeDirs := pathtools.PrefixPaths(c.Properties.Include_dirs, ctx.AConfig().SrcDir())
478 localIncludeDirs := pathtools.PrefixPaths(c.Properties.Local_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -0700479 flags.GlobalFlags = append(flags.GlobalFlags,
Dan Willemsen1e898b92015-09-23 15:26:32 -0700480 includeDirsToFlags(localIncludeDirs),
481 includeDirsToFlags(rootIncludeDirs))
Colin Cross28344522015-04-22 13:07:53 -0700482
Colin Cross39d97f22015-09-14 12:30:50 -0700483 rootIncludeFiles := pathtools.PrefixPaths(c.Properties.Include_files, ctx.AConfig().SrcDir())
484 localIncludeFiles := pathtools.PrefixPaths(c.Properties.Local_include_files, common.ModuleSrcDir(ctx))
485
486 flags.GlobalFlags = append(flags.GlobalFlags,
487 includeFilesToFlags(rootIncludeFiles),
488 includeFilesToFlags(localIncludeFiles))
489
Colin Cross06a931b2015-10-28 17:23:31 -0700490 if !Bool(c.Properties.No_default_compiler_flags) {
Colin Crossfa138792015-04-24 17:31:52 -0700491 if c.Properties.Sdk_version == "" || ctx.Host() {
Colin Cross28344522015-04-22 13:07:53 -0700492 flags.GlobalFlags = append(flags.GlobalFlags,
493 "${commonGlobalIncludes}",
494 toolchain.IncludeFlags(),
495 "-I${SrcDir}/libnativehelper/include/nativehelper")
496 }
497
498 flags.GlobalFlags = append(flags.GlobalFlags, []string{
499 "-I" + common.ModuleSrcDir(ctx),
500 "-I" + common.ModuleOutDir(ctx),
501 "-I" + common.ModuleGenDir(ctx),
502 }...)
503 }
504
Colin Cross06a931b2015-10-28 17:23:31 -0700505 if c.Properties.Clang == nil {
Dan Willemsendd0e2c32015-10-20 14:29:35 -0700506 if ctx.Host() {
507 flags.Clang = true
508 }
509
510 if ctx.Device() && ctx.AConfig().DeviceUsesClang() {
511 flags.Clang = true
512 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800513 }
514
Dan Willemsen6d11dd82015-11-03 14:27:00 -0800515 instructionSet := c.Properties.Instruction_set
516 instructionSetFlags, err := toolchain.InstructionSetFlags(instructionSet)
517 if flags.Clang {
518 instructionSetFlags, err = toolchain.ClangInstructionSetFlags(instructionSet)
519 }
520 if err != nil {
521 ctx.ModuleErrorf("%s", err)
522 }
523
524 // TODO: debug
525 flags.CFlags = append(flags.CFlags, c.Properties.Release.Cflags...)
526
Colin Cross97ba0732015-03-23 17:50:24 -0700527 if flags.Clang {
528 flags.CFlags = clangFilterUnknownCflags(flags.CFlags)
Colin Crossfa138792015-04-24 17:31:52 -0700529 flags.CFlags = append(flags.CFlags, c.Properties.Clang_cflags...)
530 flags.AsFlags = append(flags.AsFlags, c.Properties.Clang_asflags...)
Colin Cross97ba0732015-03-23 17:50:24 -0700531 flags.CppFlags = clangFilterUnknownCflags(flags.CppFlags)
532 flags.ConlyFlags = clangFilterUnknownCflags(flags.ConlyFlags)
533 flags.LdFlags = clangFilterUnknownCflags(flags.LdFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800534
Colin Cross97ba0732015-03-23 17:50:24 -0700535 flags.CFlags = append(flags.CFlags, "${clangExtraCflags}")
536 flags.ConlyFlags = append(flags.ConlyFlags, "${clangExtraConlyflags}")
Colin Crossf6566ed2015-03-24 11:13:38 -0700537 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -0700538 flags.CFlags = append(flags.CFlags, "${clangExtraTargetCflags}")
Colin Crossbdd7b1c2015-03-16 16:21:20 -0700539 }
540
Colin Cross3f40fa42015-01-30 17:27:36 -0800541 target := "-target " + toolchain.ClangTriple()
542 gccPrefix := "-B" + filepath.Join(toolchain.GccRoot(), toolchain.GccTriple(), "bin")
543
Colin Cross97ba0732015-03-23 17:50:24 -0700544 flags.CFlags = append(flags.CFlags, target, gccPrefix)
545 flags.AsFlags = append(flags.AsFlags, target, gccPrefix)
546 flags.LdFlags = append(flags.LdFlags, target, gccPrefix)
Colin Cross3f40fa42015-01-30 17:27:36 -0800547 }
548
Colin Cross06a931b2015-10-28 17:23:31 -0700549 if !Bool(c.Properties.No_default_compiler_flags) {
550 if ctx.Device() && !Bool(c.Properties.Allow_undefined_symbols) {
Colin Cross97ba0732015-03-23 17:50:24 -0700551 flags.LdFlags = append(flags.LdFlags, "-Wl,--no-undefined")
Colin Cross3f40fa42015-01-30 17:27:36 -0800552 }
553
Colin Cross56b4d452015-04-21 17:38:44 -0700554 flags.GlobalFlags = append(flags.GlobalFlags, instructionSetFlags)
555
Colin Cross97ba0732015-03-23 17:50:24 -0700556 if flags.Clang {
557 flags.CppFlags = append(flags.CppFlags, "${commonClangGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700558 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800559 toolchain.ClangCflags(),
560 "${commonClangGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700561 fmt.Sprintf("${%sClangGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800562 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700563 flags.CppFlags = append(flags.CppFlags, "${commonGlobalCppflags}")
Colin Cross56b4d452015-04-21 17:38:44 -0700564 flags.GlobalFlags = append(flags.GlobalFlags,
Colin Cross3f40fa42015-01-30 17:27:36 -0800565 toolchain.Cflags(),
566 "${commonGlobalCflags}",
Colin Crossd3ba0392015-05-07 14:11:29 -0700567 fmt.Sprintf("${%sGlobalCflags}", ctx.HostOrDevice()))
Colin Cross3f40fa42015-01-30 17:27:36 -0800568 }
569
Colin Crossf6566ed2015-03-24 11:13:38 -0700570 if ctx.Device() {
Colin Cross06a931b2015-10-28 17:23:31 -0700571 if Bool(c.Properties.Rtti) {
Colin Cross97ba0732015-03-23 17:50:24 -0700572 flags.CppFlags = append(flags.CppFlags, "-frtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800573 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700574 flags.CppFlags = append(flags.CppFlags, "-fno-rtti")
Colin Cross3f40fa42015-01-30 17:27:36 -0800575 }
576 }
577
Colin Cross97ba0732015-03-23 17:50:24 -0700578 flags.AsFlags = append(flags.AsFlags, "-D__ASSEMBLY__")
Colin Cross3f40fa42015-01-30 17:27:36 -0800579
Colin Cross97ba0732015-03-23 17:50:24 -0700580 if flags.Clang {
581 flags.CppFlags = append(flags.CppFlags, toolchain.ClangCppflags())
582 flags.LdFlags = append(flags.LdFlags, toolchain.ClangLdflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800583 } else {
Colin Cross97ba0732015-03-23 17:50:24 -0700584 flags.CppFlags = append(flags.CppFlags, toolchain.Cppflags())
585 flags.LdFlags = append(flags.LdFlags, toolchain.Ldflags())
Colin Cross3f40fa42015-01-30 17:27:36 -0800586 }
Colin Cross28344522015-04-22 13:07:53 -0700587
588 if ctx.Host() {
Colin Crossfa138792015-04-24 17:31:52 -0700589 flags.LdFlags = append(flags.LdFlags, c.Properties.Host_ldlibs...)
Colin Cross28344522015-04-22 13:07:53 -0700590 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800591 }
592
Colin Cross0676e2d2015-04-24 17:39:18 -0700593 flags = c.ccModuleType().flags(ctx, flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800594
Dan Willemsen3bf6b472015-09-11 17:41:10 -0700595 if c.Properties.Sdk_version == "" {
596 if ctx.Host() && !flags.Clang {
597 // The host GCC doesn't support C++14 (and is deprecated, so likely
598 // never will). Build these modules with C++11.
599 flags.CppFlags = append(flags.CppFlags, "-std=gnu++11")
600 } else {
601 flags.CppFlags = append(flags.CppFlags, "-std=gnu++14")
602 }
603 }
604
605 flags.CFlags, _ = filterList(flags.CFlags, illegalFlags)
606 flags.CppFlags, _ = filterList(flags.CppFlags, illegalFlags)
607 flags.ConlyFlags, _ = filterList(flags.ConlyFlags, illegalFlags)
608
Colin Cross3f40fa42015-01-30 17:27:36 -0800609 // Optimization to reduce size of build.ninja
610 // Replace the long list of flags for each file with a module-local variable
Colin Cross97ba0732015-03-23 17:50:24 -0700611 ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " "))
612 ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " "))
613 ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " "))
614 flags.CFlags = []string{"$cflags"}
615 flags.CppFlags = []string{"$cppflags"}
616 flags.AsFlags = []string{"$asflags"}
Colin Cross3f40fa42015-01-30 17:27:36 -0800617
618 return flags
619}
620
Colin Cross0676e2d2015-04-24 17:39:18 -0700621func (c *CCBase) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross3f40fa42015-01-30 17:27:36 -0800622 return flags
623}
624
625// Compile a list of source files into objects a specified subdirectory
Colin Crossfa138792015-04-24 17:31:52 -0700626func (c *CCBase) customCompileObjs(ctx common.AndroidModuleContext, flags CCFlags,
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700627 subdir string, srcFiles, excludes []string) []string {
Colin Cross581c1892015-04-07 16:50:10 -0700628
629 buildFlags := ccFlagsToBuilderFlags(flags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800630
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700631 srcFiles = ctx.ExpandSources(srcFiles, excludes)
Colin Cross581c1892015-04-07 16:50:10 -0700632 srcFiles, deps := genSources(ctx, srcFiles, buildFlags)
Colin Cross3f40fa42015-01-30 17:27:36 -0800633
Colin Cross581c1892015-04-07 16:50:10 -0700634 return TransformSourceToObj(ctx, subdir, srcFiles, buildFlags, deps)
Colin Cross3f40fa42015-01-30 17:27:36 -0800635}
636
Colin Crossfa138792015-04-24 17:31:52 -0700637// Compile files listed in c.Properties.Srcs into objects
638func (c *CCBase) compileObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800639
Colin Crossfa138792015-04-24 17:31:52 -0700640 if c.Properties.SkipCompileObjs {
Colin Cross3f40fa42015-01-30 17:27:36 -0800641 return nil
642 }
643
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700644 return c.customCompileObjs(ctx, flags, "", c.Properties.Srcs, c.Properties.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -0800645}
646
Colin Cross5049f022015-03-18 13:28:46 -0700647// Compile generated source files from dependencies
Colin Crossfa138792015-04-24 17:31:52 -0700648func (c *CCBase) compileGeneratedObjs(ctx common.AndroidModuleContext, flags CCFlags) []string {
Colin Cross5049f022015-03-18 13:28:46 -0700649 var srcs []string
650
Colin Crossfa138792015-04-24 17:31:52 -0700651 if c.Properties.SkipCompileObjs {
Colin Cross5049f022015-03-18 13:28:46 -0700652 return nil
653 }
654
655 ctx.VisitDirectDeps(func(module blueprint.Module) {
656 if gen, ok := module.(genrule.SourceFileGenerator); ok {
657 srcs = append(srcs, gen.GeneratedSourceFiles()...)
658 }
659 })
660
661 if len(srcs) == 0 {
662 return nil
663 }
664
Colin Cross581c1892015-04-07 16:50:10 -0700665 return TransformSourceToObj(ctx, "", srcs, ccFlagsToBuilderFlags(flags), nil)
Colin Cross5049f022015-03-18 13:28:46 -0700666}
667
Colin Crossfa138792015-04-24 17:31:52 -0700668func (c *CCBase) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800669 return ""
670}
671
Colin Crossfa138792015-04-24 17:31:52 -0700672func (c *CCBase) depsToPathsFromList(ctx common.AndroidModuleContext,
Colin Cross3f40fa42015-01-30 17:27:36 -0800673 names []string) (modules []common.AndroidModule,
Colin Cross28344522015-04-22 13:07:53 -0700674 outputFiles []string, exportedFlags []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800675
676 for _, n := range names {
677 found := false
678 ctx.VisitDirectDeps(func(m blueprint.Module) {
679 otherName := ctx.OtherModuleName(m)
680 if otherName != n {
681 return
682 }
683
Colin Cross97ba0732015-03-23 17:50:24 -0700684 if a, ok := m.(CCModuleType); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -0800685 if a.Disabled() {
686 // If a cc_library host+device module depends on a library that exists as both
687 // cc_library_shared and cc_library_host_shared, it will end up with two
688 // dependencies with the same name, one of which is marked disabled for each
689 // of host and device. Ignore the disabled one.
690 return
691 }
Colin Crossd3ba0392015-05-07 14:11:29 -0700692 if a.HostOrDevice() != ctx.HostOrDevice() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800693 ctx.ModuleErrorf("host/device mismatch between %q and %q", ctx.ModuleName(),
694 otherName)
695 return
696 }
697
698 if outputFile := a.outputFile(); outputFile != "" {
699 if found {
700 ctx.ModuleErrorf("multiple modules satisified dependency on %q", otherName)
701 return
702 }
703 outputFiles = append(outputFiles, outputFile)
704 modules = append(modules, a)
Colin Cross28344522015-04-22 13:07:53 -0700705 if i, ok := a.(ccExportedFlagsProducer); ok {
706 exportedFlags = append(exportedFlags, i.exportedFlags()...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800707 }
708 found = true
709 } else {
710 ctx.ModuleErrorf("module %q missing output file", otherName)
711 return
712 }
713 } else {
714 ctx.ModuleErrorf("module %q not an android module", otherName)
715 return
716 }
717 })
718 if !found {
719 ctx.ModuleErrorf("unsatisified dependency on %q", n)
720 }
721 }
722
Colin Cross28344522015-04-22 13:07:53 -0700723 return modules, outputFiles, exportedFlags
Colin Cross3f40fa42015-01-30 17:27:36 -0800724}
725
Colin Cross21b9a242015-03-24 14:15:58 -0700726// Convert depenedency names to paths. Takes a CCDeps containing names and returns a CCDeps
727// containing paths
Colin Crossfa138792015-04-24 17:31:52 -0700728func (c *CCBase) depsToPaths(ctx common.AndroidModuleContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -0700729 var depPaths CCDeps
Colin Cross28344522015-04-22 13:07:53 -0700730 var newCflags []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800731
Colin Cross21b9a242015-03-24 14:15:58 -0700732 var wholeStaticLibModules []common.AndroidModule
Colin Cross3f40fa42015-01-30 17:27:36 -0800733
Colin Cross28344522015-04-22 13:07:53 -0700734 wholeStaticLibModules, depPaths.WholeStaticLibs, newCflags =
Colin Cross21b9a242015-03-24 14:15:58 -0700735 c.depsToPathsFromList(ctx, depNames.WholeStaticLibs)
Colin Cross28344522015-04-22 13:07:53 -0700736 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Crossa48f71f2015-11-16 18:00:41 -0800737 depPaths.ReexportedCflags = append(depPaths.ReexportedCflags, newCflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800738
Colin Cross21b9a242015-03-24 14:15:58 -0700739 for _, m := range wholeStaticLibModules {
740 if staticLib, ok := m.(ccLibraryInterface); ok && staticLib.static() {
741 depPaths.WholeStaticLibObjFiles =
742 append(depPaths.WholeStaticLibObjFiles, staticLib.allObjFiles()...)
743 } else {
744 ctx.ModuleErrorf("module %q not a static library", ctx.OtherModuleName(m))
745 }
746 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800747
Colin Cross28344522015-04-22 13:07:53 -0700748 _, depPaths.StaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.StaticLibs)
749 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700750
Colin Cross28344522015-04-22 13:07:53 -0700751 _, depPaths.LateStaticLibs, newCflags = c.depsToPathsFromList(ctx, depNames.LateStaticLibs)
752 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700753
Colin Cross28344522015-04-22 13:07:53 -0700754 _, depPaths.SharedLibs, newCflags = c.depsToPathsFromList(ctx, depNames.SharedLibs)
755 depPaths.Cflags = append(depPaths.Cflags, newCflags...)
Colin Cross21b9a242015-03-24 14:15:58 -0700756
757 ctx.VisitDirectDeps(func(m blueprint.Module) {
Dan Albertc3144b12015-04-28 18:17:56 -0700758 if obj, ok := m.(ccObjectProvider); ok {
Colin Cross21b9a242015-03-24 14:15:58 -0700759 otherName := ctx.OtherModuleName(m)
760 if otherName == depNames.CrtBegin {
Colin Cross06a931b2015-10-28 17:23:31 -0700761 if !Bool(c.Properties.Nocrt) {
Dan Albertc3144b12015-04-28 18:17:56 -0700762 depPaths.CrtBegin = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700763 }
764 } else if otherName == depNames.CrtEnd {
Colin Cross06a931b2015-10-28 17:23:31 -0700765 if !Bool(c.Properties.Nocrt) {
Dan Albertc3144b12015-04-28 18:17:56 -0700766 depPaths.CrtEnd = obj.object().outputFile()
Colin Cross21b9a242015-03-24 14:15:58 -0700767 }
768 } else {
Dan Albertc3144b12015-04-28 18:17:56 -0700769 depPaths.ObjFiles = append(depPaths.ObjFiles, obj.object().outputFile())
Colin Cross21b9a242015-03-24 14:15:58 -0700770 }
771 }
772 })
773
774 return depPaths
Colin Cross3f40fa42015-01-30 17:27:36 -0800775}
776
Colin Cross7d5136f2015-05-11 13:39:40 -0700777type ccLinkedProperties struct {
778 VariantIsShared bool `blueprint:"mutated"`
779 VariantIsStatic bool `blueprint:"mutated"`
780 VariantIsStaticBinary bool `blueprint:"mutated"`
781}
782
Colin Crossfa138792015-04-24 17:31:52 -0700783// CCLinked contains the properties and members used by libraries and executables
784type CCLinked struct {
785 CCBase
Colin Cross7d5136f2015-05-11 13:39:40 -0700786 dynamicProperties ccLinkedProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800787}
788
Colin Crossfa138792015-04-24 17:31:52 -0700789func newCCDynamic(dynamic *CCLinked, module CCModuleType, hod common.HostOrDeviceSupported,
Colin Crossc472d572015-03-17 15:06:21 -0700790 multilib common.Multilib, props ...interface{}) (blueprint.Module, []interface{}) {
791
Colin Crossed4cf0b2015-03-26 14:43:45 -0700792 props = append(props, &dynamic.dynamicProperties)
793
Colin Crossfa138792015-04-24 17:31:52 -0700794 return newCCBase(&dynamic.CCBase, module, hod, multilib, props...)
Colin Crossc472d572015-03-17 15:06:21 -0700795}
796
Colin Crossfa138792015-04-24 17:31:52 -0700797func (c *CCLinked) systemSharedLibs(ctx common.AndroidBaseContext) []string {
Colin Cross06a931b2015-10-28 17:23:31 -0700798 if c.Properties.System_shared_libs != nil {
Colin Crossfa138792015-04-24 17:31:52 -0700799 return c.Properties.System_shared_libs
800 } else if ctx.Device() && c.Properties.Sdk_version == "" {
Colin Cross577f6e42015-03-27 18:23:34 -0700801 return []string{"libc", "libm"}
Colin Cross28d76592015-03-26 16:14:04 -0700802 } else {
Colin Cross577f6e42015-03-27 18:23:34 -0700803 return nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800804 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800805}
806
Colin Crossfa138792015-04-24 17:31:52 -0700807func (c *CCLinked) stl(ctx common.AndroidBaseContext) string {
808 if c.Properties.Sdk_version != "" && ctx.Device() {
809 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700810 case "":
811 return "ndk_system"
812 case "c++_shared", "c++_static",
813 "stlport_shared", "stlport_static",
814 "gnustl_static":
Colin Crossfa138792015-04-24 17:31:52 -0700815 return "ndk_lib" + c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700816 default:
Colin Crossfa138792015-04-24 17:31:52 -0700817 ctx.ModuleErrorf("stl: %q is not a supported STL with sdk_version set", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700818 return ""
819 }
820 }
821
Colin Crossfa138792015-04-24 17:31:52 -0700822 switch c.Properties.Stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700823 case "libc++", "libc++_static",
Colin Crossed4cf0b2015-03-26 14:43:45 -0700824 "libstdc++":
Colin Crossfa138792015-04-24 17:31:52 -0700825 return c.Properties.Stl
Colin Crossed4cf0b2015-03-26 14:43:45 -0700826 case "none":
827 return ""
828 case "":
Colin Cross18b6dc52015-04-28 13:20:37 -0700829 if c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700830 return "libc++_static"
Colin Cross18b6dc52015-04-28 13:20:37 -0700831 } else {
832 return "libc++" // TODO: mingw needs libstdc++
Colin Crossed4cf0b2015-03-26 14:43:45 -0700833 }
834 default:
Colin Crossfa138792015-04-24 17:31:52 -0700835 ctx.ModuleErrorf("stl: %q is not a supported STL", c.Properties.Stl)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700836 return ""
837 }
838}
839
Colin Cross0af4b842015-04-30 16:36:18 -0700840var hostDynamicGccLibs, hostStaticGccLibs []string
841
842func init() {
843 if runtime.GOOS == "darwin" {
844 hostDynamicGccLibs = []string{"-lc", "-lSystem"}
845 hostStaticGccLibs = []string{"NO_STATIC_HOST_BINARIES_ON_DARWIN"}
846 } else {
847 hostDynamicGccLibs = []string{"-lgcc_s", "-lgcc", "-lc", "-lgcc_s", "-lgcc"}
848 hostStaticGccLibs = []string{"-Wl,--start-group", "-lgcc", "-lgcc_eh", "-lc", "-Wl,--end-group"}
849 }
850}
Colin Cross712fc022015-04-27 11:13:34 -0700851
Colin Crosse11befc2015-04-27 17:49:17 -0700852func (c *CCLinked) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700853 stl := c.stl(ctx)
854 if ctx.Failed() {
855 return flags
856 }
857
858 switch stl {
859 case "libc++", "libc++_static":
860 flags.CFlags = append(flags.CFlags, "-D_USING_LIBCXX")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700861 if ctx.Host() {
862 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
863 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross712fc022015-04-27 11:13:34 -0700864 flags.LdFlags = append(flags.LdFlags, "-lm", "-lpthread")
Colin Cross18b6dc52015-04-28 13:20:37 -0700865 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700866 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700867 } else {
868 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700869 }
Dan Willemsen3bf6b472015-09-11 17:41:10 -0700870 } else {
871 if ctx.Arch().ArchType == common.Arm {
872 flags.LdFlags = append(flags.LdFlags, "-Wl,--exclude-libs,libunwind_llvm.a")
873 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700874 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700875 case "libstdc++":
876 // Using bionic's basic libstdc++. Not actually an STL. Only around until the
877 // tree is in good enough shape to not need it.
878 // Host builds will use GNU libstdc++.
879 if ctx.Device() {
Colin Cross28344522015-04-22 13:07:53 -0700880 flags.CFlags = append(flags.CFlags, "-I${SrcDir}/bionic/libstdc++/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700881 }
882 case "ndk_system":
Colin Cross1332b002015-04-07 17:11:30 -0700883 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources/"
Colin Cross28344522015-04-22 13:07:53 -0700884 flags.CFlags = append(flags.CFlags, "-isystem "+ndkSrcRoot+"cxx-stl/system/include")
Colin Crossed4cf0b2015-03-26 14:43:45 -0700885 case "ndk_libc++_shared", "ndk_libc++_static":
886 // TODO(danalbert): This really shouldn't be here...
887 flags.CppFlags = append(flags.CppFlags, "-std=c++11")
888 case "ndk_libstlport_shared", "ndk_libstlport_static", "ndk_libgnustl_static":
889 // Nothing
890 case "":
891 // None or error.
892 if ctx.Host() {
893 flags.CppFlags = append(flags.CppFlags, "-nostdinc++")
894 flags.LdFlags = append(flags.LdFlags, "-nodefaultlibs")
Colin Cross18b6dc52015-04-28 13:20:37 -0700895 if c.staticBinary() {
Colin Cross712fc022015-04-27 11:13:34 -0700896 flags.LdFlags = append(flags.LdFlags, hostStaticGccLibs...)
Colin Cross18b6dc52015-04-28 13:20:37 -0700897 } else {
898 flags.LdFlags = append(flags.LdFlags, hostDynamicGccLibs...)
Colin Cross712fc022015-04-27 11:13:34 -0700899 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700900 }
901 default:
Colin Crossfa138792015-04-24 17:31:52 -0700902 panic(fmt.Errorf("Unknown stl in CCLinked.Flags: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700903 }
904
905 return flags
906}
907
Colin Crosse11befc2015-04-27 17:49:17 -0700908func (c *CCLinked) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
909 depNames = c.CCBase.depNames(ctx, depNames)
Colin Cross3f40fa42015-01-30 17:27:36 -0800910
Colin Crossed4cf0b2015-03-26 14:43:45 -0700911 stl := c.stl(ctx)
912 if ctx.Failed() {
913 return depNames
914 }
915
916 switch stl {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700917 case "libstdc++":
918 if ctx.Device() {
919 depNames.SharedLibs = append(depNames.SharedLibs, stl)
920 }
Colin Cross74d1ec02015-04-28 13:30:13 -0700921 case "libc++", "libc++_static":
922 if stl == "libc++" {
923 depNames.SharedLibs = append(depNames.SharedLibs, stl)
924 } else {
925 depNames.StaticLibs = append(depNames.StaticLibs, stl)
926 }
927 if ctx.Device() {
928 if ctx.Arch().ArchType == common.Arm {
929 depNames.StaticLibs = append(depNames.StaticLibs, "libunwind_llvm")
930 }
931 if c.staticBinary() {
932 depNames.StaticLibs = append(depNames.StaticLibs, "libdl")
933 } else {
934 depNames.SharedLibs = append(depNames.SharedLibs, "libdl")
935 }
936 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700937 case "":
938 // None or error.
939 case "ndk_system":
940 // TODO: Make a system STL prebuilt for the NDK.
941 // The system STL doesn't have a prebuilt (it uses the system's libstdc++), but it does have
Colin Crossfa138792015-04-24 17:31:52 -0700942 // its own includes. The includes are handled in CCBase.Flags().
Colin Cross577f6e42015-03-27 18:23:34 -0700943 depNames.SharedLibs = append([]string{"libstdc++"}, depNames.SharedLibs...)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700944 case "ndk_libc++_shared", "ndk_libstlport_shared":
945 depNames.SharedLibs = append(depNames.SharedLibs, stl)
946 case "ndk_libc++_static", "ndk_libstlport_static", "ndk_libgnustl_static":
947 depNames.StaticLibs = append(depNames.StaticLibs, stl)
948 default:
Colin Crosse11befc2015-04-27 17:49:17 -0700949 panic(fmt.Errorf("Unknown stl in CCLinked.depNames: %q", stl))
Colin Crossed4cf0b2015-03-26 14:43:45 -0700950 }
951
Colin Cross74d1ec02015-04-28 13:30:13 -0700952 if ctx.ModuleName() != "libcompiler_rt-extras" {
953 depNames.StaticLibs = append(depNames.StaticLibs, "libcompiler_rt-extras")
954 }
955
Colin Crossf6566ed2015-03-24 11:13:38 -0700956 if ctx.Device() {
Colin Cross77b00fa2015-03-16 16:15:49 -0700957 // libgcc and libatomic have to be last on the command line
Dan Willemsend67be222015-09-16 15:19:33 -0700958 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcov", "libatomic")
Colin Cross06a931b2015-10-28 17:23:31 -0700959 if !Bool(c.Properties.No_libgcc) {
Dan Willemsend67be222015-09-16 15:19:33 -0700960 depNames.LateStaticLibs = append(depNames.LateStaticLibs, "libgcc")
961 }
Colin Crossed4cf0b2015-03-26 14:43:45 -0700962
Colin Cross18b6dc52015-04-28 13:20:37 -0700963 if !c.static() {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700964 depNames.SharedLibs = append(depNames.SharedLibs, c.systemSharedLibs(ctx)...)
965 }
Colin Cross577f6e42015-03-27 18:23:34 -0700966
Colin Crossfa138792015-04-24 17:31:52 -0700967 if c.Properties.Sdk_version != "" {
968 version := c.Properties.Sdk_version
Colin Cross577f6e42015-03-27 18:23:34 -0700969 depNames.SharedLibs = append(depNames.SharedLibs,
970 "ndk_libc."+version,
971 "ndk_libm."+version,
972 )
973 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800974 }
975
Colin Cross21b9a242015-03-24 14:15:58 -0700976 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -0800977}
978
Colin Crossed4cf0b2015-03-26 14:43:45 -0700979// ccLinkedInterface interface is used on ccLinked to deal with static or shared variants
980type ccLinkedInterface interface {
981 // Returns true if the build options for the module have selected a static or shared build
982 buildStatic() bool
983 buildShared() bool
984
985 // Sets whether a specific variant is static or shared
Colin Cross18b6dc52015-04-28 13:20:37 -0700986 setStatic(bool)
Colin Crossed4cf0b2015-03-26 14:43:45 -0700987
Colin Cross18b6dc52015-04-28 13:20:37 -0700988 // Returns whether a specific variant is a static library or binary
Colin Crossed4cf0b2015-03-26 14:43:45 -0700989 static() bool
Colin Cross18b6dc52015-04-28 13:20:37 -0700990
991 // Returns whether a module is a static binary
992 staticBinary() bool
Colin Crossed4cf0b2015-03-26 14:43:45 -0700993}
994
995var _ ccLinkedInterface = (*CCLibrary)(nil)
996var _ ccLinkedInterface = (*CCBinary)(nil)
997
Colin Crossfa138792015-04-24 17:31:52 -0700998func (c *CCLinked) static() bool {
Colin Crossed4cf0b2015-03-26 14:43:45 -0700999 return c.dynamicProperties.VariantIsStatic
1000}
1001
Colin Cross18b6dc52015-04-28 13:20:37 -07001002func (c *CCLinked) staticBinary() bool {
1003 return c.dynamicProperties.VariantIsStaticBinary
Colin Crossed4cf0b2015-03-26 14:43:45 -07001004}
1005
Colin Cross18b6dc52015-04-28 13:20:37 -07001006func (c *CCLinked) setStatic(static bool) {
1007 c.dynamicProperties.VariantIsStatic = static
Colin Crossed4cf0b2015-03-26 14:43:45 -07001008}
1009
Colin Cross28344522015-04-22 13:07:53 -07001010type ccExportedFlagsProducer interface {
1011 exportedFlags() []string
Colin Cross3f40fa42015-01-30 17:27:36 -08001012}
1013
1014//
1015// Combined static+shared libraries
1016//
1017
Colin Cross7d5136f2015-05-11 13:39:40 -07001018type CCLibraryProperties struct {
1019 BuildStatic bool `blueprint:"mutated"`
1020 BuildShared bool `blueprint:"mutated"`
1021 Static struct {
1022 Srcs []string `android:"arch_variant"`
Dan Willemsen2ef08f42015-06-30 18:15:24 -07001023 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -07001024 Cflags []string `android:"arch_variant"`
1025 Whole_static_libs []string `android:"arch_variant"`
1026 Static_libs []string `android:"arch_variant"`
1027 Shared_libs []string `android:"arch_variant"`
1028 } `android:"arch_variant"`
1029 Shared struct {
1030 Srcs []string `android:"arch_variant"`
Dan Willemsen2ef08f42015-06-30 18:15:24 -07001031 Exclude_srcs []string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -07001032 Cflags []string `android:"arch_variant"`
1033 Whole_static_libs []string `android:"arch_variant"`
1034 Static_libs []string `android:"arch_variant"`
1035 Shared_libs []string `android:"arch_variant"`
1036 } `android:"arch_variant"`
Colin Crossaee540a2015-07-06 17:48:31 -07001037
1038 // local file name to pass to the linker as --version_script
1039 Version_script string `android:"arch_variant"`
Colin Cross7d5136f2015-05-11 13:39:40 -07001040}
1041
Colin Cross97ba0732015-03-23 17:50:24 -07001042type CCLibrary struct {
Colin Crossfa138792015-04-24 17:31:52 -07001043 CCLinked
Colin Cross3f40fa42015-01-30 17:27:36 -08001044
Colin Cross28344522015-04-22 13:07:53 -07001045 reuseFrom ccLibraryInterface
1046 reuseObjFiles []string
1047 objFiles []string
1048 exportFlags []string
1049 out string
Colin Cross3f40fa42015-01-30 17:27:36 -08001050
Colin Cross7d5136f2015-05-11 13:39:40 -07001051 LibraryProperties CCLibraryProperties
Colin Cross3f40fa42015-01-30 17:27:36 -08001052}
1053
Colin Crossed4cf0b2015-03-26 14:43:45 -07001054func (c *CCLibrary) buildStatic() bool {
1055 return c.LibraryProperties.BuildStatic
1056}
1057
1058func (c *CCLibrary) buildShared() bool {
1059 return c.LibraryProperties.BuildShared
1060}
1061
Colin Cross97ba0732015-03-23 17:50:24 -07001062type ccLibraryInterface interface {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001063 ccLinkedInterface
Colin Cross97ba0732015-03-23 17:50:24 -07001064 ccLibrary() *CCLibrary
Colin Crossed4cf0b2015-03-26 14:43:45 -07001065 setReuseFrom(ccLibraryInterface)
1066 getReuseFrom() ccLibraryInterface
1067 getReuseObjFiles() []string
Colin Cross97ba0732015-03-23 17:50:24 -07001068 allObjFiles() []string
Colin Crossc472d572015-03-17 15:06:21 -07001069}
1070
Colin Crossed4cf0b2015-03-26 14:43:45 -07001071var _ ccLibraryInterface = (*CCLibrary)(nil)
1072
Colin Cross97ba0732015-03-23 17:50:24 -07001073func (c *CCLibrary) ccLibrary() *CCLibrary {
1074 return c
Colin Cross3f40fa42015-01-30 17:27:36 -08001075}
1076
Colin Cross97ba0732015-03-23 17:50:24 -07001077func NewCCLibrary(library *CCLibrary, module CCModuleType,
1078 hod common.HostOrDeviceSupported) (blueprint.Module, []interface{}) {
1079
Colin Crossfa138792015-04-24 17:31:52 -07001080 return newCCDynamic(&library.CCLinked, module, hod, common.MultilibBoth,
Colin Cross97ba0732015-03-23 17:50:24 -07001081 &library.LibraryProperties)
1082}
1083
1084func CCLibraryFactory() (blueprint.Module, []interface{}) {
1085 module := &CCLibrary{}
1086
1087 module.LibraryProperties.BuildShared = true
1088 module.LibraryProperties.BuildStatic = true
1089
1090 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
1091}
1092
Colin Cross0676e2d2015-04-24 17:39:18 -07001093func (c *CCLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001094 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Cross2732e9a2015-04-28 13:23:52 -07001095 if c.static() {
1096 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Static.Whole_static_libs...)
1097 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Static.Static_libs...)
1098 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Static.Shared_libs...)
1099 } else {
Colin Crossf6566ed2015-03-24 11:13:38 -07001100 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001101 if c.Properties.Sdk_version == "" {
1102 depNames.CrtBegin = "crtbegin_so"
1103 depNames.CrtEnd = "crtend_so"
1104 } else {
1105 depNames.CrtBegin = "ndk_crtbegin_so." + c.Properties.Sdk_version
1106 depNames.CrtEnd = "ndk_crtend_so." + c.Properties.Sdk_version
1107 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001108 }
Colin Cross2732e9a2015-04-28 13:23:52 -07001109 depNames.WholeStaticLibs = append(depNames.WholeStaticLibs, c.LibraryProperties.Shared.Whole_static_libs...)
1110 depNames.StaticLibs = append(depNames.StaticLibs, c.LibraryProperties.Shared.Static_libs...)
1111 depNames.SharedLibs = append(depNames.SharedLibs, c.LibraryProperties.Shared.Shared_libs...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001112 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001113
Colin Cross21b9a242015-03-24 14:15:58 -07001114 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001115}
1116
Colin Cross97ba0732015-03-23 17:50:24 -07001117func (c *CCLibrary) outputFile() string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001118 return c.out
1119}
1120
Colin Crossed4cf0b2015-03-26 14:43:45 -07001121func (c *CCLibrary) getReuseObjFiles() []string {
1122 return c.reuseObjFiles
1123}
1124
1125func (c *CCLibrary) setReuseFrom(reuseFrom ccLibraryInterface) {
1126 c.reuseFrom = reuseFrom
1127}
1128
1129func (c *CCLibrary) getReuseFrom() ccLibraryInterface {
1130 return c.reuseFrom
1131}
1132
Colin Cross97ba0732015-03-23 17:50:24 -07001133func (c *CCLibrary) allObjFiles() []string {
Colin Cross3f40fa42015-01-30 17:27:36 -08001134 return c.objFiles
1135}
1136
Colin Cross28344522015-04-22 13:07:53 -07001137func (c *CCLibrary) exportedFlags() []string {
1138 return c.exportFlags
Colin Cross3f40fa42015-01-30 17:27:36 -08001139}
1140
Colin Cross0676e2d2015-04-24 17:39:18 -07001141func (c *CCLibrary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001142 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001143
Colin Cross97ba0732015-03-23 17:50:24 -07001144 flags.CFlags = append(flags.CFlags, "-fPIC")
Colin Cross3f40fa42015-01-30 17:27:36 -08001145
Colin Crossd8e780d2015-04-28 17:39:43 -07001146 if c.static() {
1147 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Static.Cflags...)
1148 } else {
1149 flags.CFlags = append(flags.CFlags, c.LibraryProperties.Shared.Cflags...)
1150 }
1151
Colin Cross18b6dc52015-04-28 13:20:37 -07001152 if !c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001153 libName := ctx.ModuleName()
1154 // GCC for Android assumes that -shared means -Bsymbolic, use -Wl,-shared instead
1155 sharedFlag := "-Wl,-shared"
Dan Willemsendd0e2c32015-10-20 14:29:35 -07001156 if flags.Clang || ctx.Host() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001157 sharedFlag = "-shared"
1158 }
Colin Crossf6566ed2015-03-24 11:13:38 -07001159 if ctx.Device() {
Colin Cross97ba0732015-03-23 17:50:24 -07001160 flags.LdFlags = append(flags.LdFlags, "-nostdlib")
Colin Cross3f40fa42015-01-30 17:27:36 -08001161 }
Colin Cross97ba0732015-03-23 17:50:24 -07001162
Colin Cross0af4b842015-04-30 16:36:18 -07001163 if ctx.Darwin() {
1164 flags.LdFlags = append(flags.LdFlags,
1165 "-dynamiclib",
1166 "-single_module",
1167 //"-read_only_relocs suppress",
1168 "-install_name @rpath/"+libName+sharedLibraryExtension,
1169 )
1170 } else {
1171 flags.LdFlags = append(flags.LdFlags,
1172 "-Wl,--gc-sections",
1173 sharedFlag,
1174 "-Wl,-soname,"+libName+sharedLibraryExtension,
1175 )
1176 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001177 }
Colin Cross97ba0732015-03-23 17:50:24 -07001178
1179 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001180}
1181
Colin Cross97ba0732015-03-23 17:50:24 -07001182func (c *CCLibrary) compileStaticLibrary(ctx common.AndroidModuleContext,
1183 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001184
1185 staticFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001186 objFilesStatic := c.customCompileObjs(ctx, staticFlags, common.DeviceStaticLibrary,
Dan Willemsen2ef08f42015-06-30 18:15:24 -07001187 c.LibraryProperties.Static.Srcs, c.LibraryProperties.Static.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001188
1189 objFiles = append(objFiles, objFilesStatic...)
Colin Cross21b9a242015-03-24 14:15:58 -07001190 objFiles = append(objFiles, deps.WholeStaticLibObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001191
1192 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+staticLibraryExtension)
1193
Colin Cross0af4b842015-04-30 16:36:18 -07001194 if ctx.Darwin() {
1195 TransformDarwinObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1196 } else {
1197 TransformObjToStaticLib(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1198 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001199
1200 c.objFiles = objFiles
1201 c.out = outputFile
Colin Crossf2298272015-05-12 11:36:53 -07001202
1203 common.CheckModuleSrcDirsExist(ctx, c.Properties.Export_include_dirs, "export_include_dirs")
Colin Crossfa138792015-04-24 17:31:52 -07001204 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001205 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Crossa48f71f2015-11-16 18:00:41 -08001206 c.exportFlags = append(c.exportFlags, deps.ReexportedCflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001207
1208 ctx.CheckbuildFile(outputFile)
1209}
1210
Colin Cross97ba0732015-03-23 17:50:24 -07001211func (c *CCLibrary) compileSharedLibrary(ctx common.AndroidModuleContext,
1212 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001213
1214 sharedFlags := flags
Colin Cross581c1892015-04-07 16:50:10 -07001215 objFilesShared := c.customCompileObjs(ctx, sharedFlags, common.DeviceSharedLibrary,
Dan Willemsen2ef08f42015-06-30 18:15:24 -07001216 c.LibraryProperties.Shared.Srcs, c.LibraryProperties.Shared.Exclude_srcs)
Colin Cross3f40fa42015-01-30 17:27:36 -08001217
1218 objFiles = append(objFiles, objFilesShared...)
1219
1220 outputFile := filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+sharedLibraryExtension)
1221
Colin Crossaee540a2015-07-06 17:48:31 -07001222 var linkerDeps []string
1223
1224 if c.LibraryProperties.Version_script != "" {
1225 versionScript := filepath.Join(common.ModuleSrcDir(ctx), c.LibraryProperties.Version_script)
1226 sharedFlags.LdFlags = append(sharedFlags.LdFlags, "-Wl,--version-script,"+versionScript)
1227 linkerDeps = append(linkerDeps, versionScript)
1228 }
1229
Colin Cross97ba0732015-03-23 17:50:24 -07001230 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossaee540a2015-07-06 17:48:31 -07001231 deps.LateStaticLibs, deps.WholeStaticLibs, linkerDeps, deps.CrtBegin, deps.CrtEnd, false,
Dan Willemsen6203ac02015-11-24 12:58:57 -08001232 ccFlagsToBuilderFlags(sharedFlags), outputFile)
Colin Cross3f40fa42015-01-30 17:27:36 -08001233
1234 c.out = outputFile
Colin Crossfa138792015-04-24 17:31:52 -07001235 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001236 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Colin Crossa48f71f2015-11-16 18:00:41 -08001237 c.exportFlags = append(c.exportFlags, deps.ReexportedCflags...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001238}
1239
Colin Cross97ba0732015-03-23 17:50:24 -07001240func (c *CCLibrary) compileModule(ctx common.AndroidModuleContext,
1241 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001242
1243 // Reuse the object files from the matching static library if it exists
Colin Crossed4cf0b2015-03-26 14:43:45 -07001244 if c.getReuseFrom().ccLibrary() == c {
1245 c.reuseObjFiles = objFiles
Colin Cross3f40fa42015-01-30 17:27:36 -08001246 } else {
Colin Cross2732e9a2015-04-28 13:23:52 -07001247 if c.getReuseFrom().ccLibrary().LibraryProperties.Static.Cflags == nil &&
1248 c.LibraryProperties.Shared.Cflags == nil {
1249 objFiles = append([]string(nil), c.getReuseFrom().getReuseObjFiles()...)
1250 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001251 }
1252
Colin Crossed4cf0b2015-03-26 14:43:45 -07001253 if c.static() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001254 c.compileStaticLibrary(ctx, flags, deps, objFiles)
1255 } else {
1256 c.compileSharedLibrary(ctx, flags, deps, objFiles)
1257 }
1258}
1259
Colin Cross97ba0732015-03-23 17:50:24 -07001260func (c *CCLibrary) installStaticLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001261 // Static libraries do not get installed.
1262}
1263
Colin Cross97ba0732015-03-23 17:50:24 -07001264func (c *CCLibrary) installSharedLibrary(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001265 installDir := "lib"
Colin Cross97ba0732015-03-23 17:50:24 -07001266 if flags.Toolchain.Is64Bit() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001267 installDir = "lib64"
1268 }
1269
Colin Crossfa138792015-04-24 17:31:52 -07001270 ctx.InstallFile(filepath.Join(installDir, c.Properties.Relative_install_path), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001271}
1272
Colin Cross97ba0732015-03-23 17:50:24 -07001273func (c *CCLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001274 if c.static() {
Dan Albertc403f7c2015-03-18 14:01:18 -07001275 c.installStaticLibrary(ctx, flags)
1276 } else {
1277 c.installSharedLibrary(ctx, flags)
1278 }
1279}
1280
Colin Cross3f40fa42015-01-30 17:27:36 -08001281//
1282// Objects (for crt*.o)
1283//
1284
Dan Albertc3144b12015-04-28 18:17:56 -07001285type ccObjectProvider interface {
1286 object() *ccObject
1287}
1288
Colin Cross3f40fa42015-01-30 17:27:36 -08001289type ccObject struct {
Colin Crossfa138792015-04-24 17:31:52 -07001290 CCBase
Colin Cross3f40fa42015-01-30 17:27:36 -08001291 out string
1292}
1293
Dan Albertc3144b12015-04-28 18:17:56 -07001294func (c *ccObject) object() *ccObject {
1295 return c
1296}
1297
Colin Cross97ba0732015-03-23 17:50:24 -07001298func CCObjectFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001299 module := &ccObject{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001300
Colin Crossfa138792015-04-24 17:31:52 -07001301 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
Colin Cross3f40fa42015-01-30 17:27:36 -08001302}
1303
Colin Cross0676e2d2015-04-24 17:39:18 -07001304func (*ccObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross21b9a242015-03-24 14:15:58 -07001305 // object files can't have any dynamic dependencies
1306 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001307}
1308
1309func (c *ccObject) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001310 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001311
Colin Cross97ba0732015-03-23 17:50:24 -07001312 objFiles = append(objFiles, deps.ObjFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001313
1314 var outputFile string
1315 if len(objFiles) == 1 {
1316 outputFile = objFiles[0]
1317 } else {
Dan Albertc3144b12015-04-28 18:17:56 -07001318 outputFile = filepath.Join(common.ModuleOutDir(ctx), ctx.ModuleName()+objectExtension)
Colin Cross3f40fa42015-01-30 17:27:36 -08001319 TransformObjsToObj(ctx, objFiles, ccFlagsToBuilderFlags(flags), outputFile)
1320 }
1321
1322 c.out = outputFile
1323
1324 ctx.CheckbuildFile(outputFile)
1325}
1326
Colin Cross97ba0732015-03-23 17:50:24 -07001327func (c *ccObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001328 // Object files do not get installed.
1329}
1330
Colin Cross3f40fa42015-01-30 17:27:36 -08001331func (c *ccObject) outputFile() string {
1332 return c.out
1333}
1334
Dan Albertc3144b12015-04-28 18:17:56 -07001335var _ ccObjectProvider = (*ccObject)(nil)
1336
Colin Cross3f40fa42015-01-30 17:27:36 -08001337//
1338// Executables
1339//
1340
Colin Cross7d5136f2015-05-11 13:39:40 -07001341type CCBinaryProperties struct {
1342 // compile executable with -static
Colin Cross06a931b2015-10-28 17:23:31 -07001343 Static_executable *bool
Colin Cross7d5136f2015-05-11 13:39:40 -07001344
1345 // set the name of the output
1346 Stem string `android:"arch_variant"`
1347
1348 // append to the name of the output
1349 Suffix string `android:"arch_variant"`
1350
1351 // if set, add an extra objcopy --prefix-symbols= step
1352 Prefix_symbols string
Colin Cross6002e052015-09-16 16:00:08 -07001353
1354 // Create a separate binary for each source file. Useful when there is
1355 // global state that can not be torn down and reset between each test suite.
Colin Cross06a931b2015-10-28 17:23:31 -07001356 Test_per_src *bool
Colin Cross7d5136f2015-05-11 13:39:40 -07001357}
1358
Colin Cross97ba0732015-03-23 17:50:24 -07001359type CCBinary struct {
Colin Crossfa138792015-04-24 17:31:52 -07001360 CCLinked
Dan Albertc403f7c2015-03-18 14:01:18 -07001361 out string
Colin Crossd350ecd2015-04-28 13:25:36 -07001362 installFile string
Colin Cross7d5136f2015-05-11 13:39:40 -07001363 BinaryProperties CCBinaryProperties
Colin Cross3f40fa42015-01-30 17:27:36 -08001364}
1365
Colin Crossed4cf0b2015-03-26 14:43:45 -07001366func (c *CCBinary) buildStatic() bool {
Colin Cross06a931b2015-10-28 17:23:31 -07001367 return Bool(c.BinaryProperties.Static_executable)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001368}
1369
1370func (c *CCBinary) buildShared() bool {
Colin Cross06a931b2015-10-28 17:23:31 -07001371 return !Bool(c.BinaryProperties.Static_executable)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001372}
1373
Colin Cross97ba0732015-03-23 17:50:24 -07001374func (c *CCBinary) getStem(ctx common.AndroidModuleContext) string {
Colin Cross4ae185c2015-03-26 15:12:10 -07001375 stem := ctx.ModuleName()
Colin Cross97ba0732015-03-23 17:50:24 -07001376 if c.BinaryProperties.Stem != "" {
Colin Cross4ae185c2015-03-26 15:12:10 -07001377 stem = c.BinaryProperties.Stem
Colin Cross3f40fa42015-01-30 17:27:36 -08001378 }
Colin Cross4ae185c2015-03-26 15:12:10 -07001379
1380 return stem + c.BinaryProperties.Suffix
Colin Cross3f40fa42015-01-30 17:27:36 -08001381}
1382
Colin Cross0676e2d2015-04-24 17:39:18 -07001383func (c *CCBinary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Crosse11befc2015-04-27 17:49:17 -07001384 depNames = c.CCLinked.depNames(ctx, depNames)
Colin Crossf6566ed2015-03-24 11:13:38 -07001385 if ctx.Device() {
Dan Albertc3144b12015-04-28 18:17:56 -07001386 if c.Properties.Sdk_version == "" {
Colin Cross06a931b2015-10-28 17:23:31 -07001387 if Bool(c.BinaryProperties.Static_executable) {
Dan Albertc3144b12015-04-28 18:17:56 -07001388 depNames.CrtBegin = "crtbegin_static"
1389 } else {
1390 depNames.CrtBegin = "crtbegin_dynamic"
1391 }
1392 depNames.CrtEnd = "crtend_android"
Colin Cross3f40fa42015-01-30 17:27:36 -08001393 } else {
Colin Cross06a931b2015-10-28 17:23:31 -07001394 if Bool(c.BinaryProperties.Static_executable) {
Dan Albertc3144b12015-04-28 18:17:56 -07001395 depNames.CrtBegin = "ndk_crtbegin_static." + c.Properties.Sdk_version
1396 } else {
1397 depNames.CrtBegin = "ndk_crtbegin_dynamic." + c.Properties.Sdk_version
1398 }
1399 depNames.CrtEnd = "ndk_crtend_android." + c.Properties.Sdk_version
Colin Cross3f40fa42015-01-30 17:27:36 -08001400 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001401
Colin Cross06a931b2015-10-28 17:23:31 -07001402 if Bool(c.BinaryProperties.Static_executable) {
Colin Cross74d1ec02015-04-28 13:30:13 -07001403 if c.stl(ctx) == "libc++_static" {
1404 depNames.StaticLibs = append(depNames.StaticLibs, "libm", "libc", "libdl")
1405 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001406 // static libraries libcompiler_rt, libc and libc_nomalloc need to be linked with
1407 // --start-group/--end-group along with libgcc. If they are in deps.StaticLibs,
1408 // move them to the beginning of deps.LateStaticLibs
1409 var groupLibs []string
1410 depNames.StaticLibs, groupLibs = filterList(depNames.StaticLibs,
1411 []string{"libc", "libc_nomalloc", "libcompiler_rt"})
1412 depNames.LateStaticLibs = append(groupLibs, depNames.LateStaticLibs...)
1413 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001414 }
Colin Cross21b9a242015-03-24 14:15:58 -07001415 return depNames
Colin Cross3f40fa42015-01-30 17:27:36 -08001416}
1417
Colin Cross97ba0732015-03-23 17:50:24 -07001418func NewCCBinary(binary *CCBinary, module CCModuleType,
Colin Cross1f8f2342015-03-26 16:09:47 -07001419 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001420
Colin Cross1f8f2342015-03-26 16:09:47 -07001421 props = append(props, &binary.BinaryProperties)
1422
Colin Crossfa138792015-04-24 17:31:52 -07001423 return newCCDynamic(&binary.CCLinked, module, hod, common.MultilibFirst, props...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001424}
1425
Colin Cross97ba0732015-03-23 17:50:24 -07001426func CCBinaryFactory() (blueprint.Module, []interface{}) {
1427 module := &CCBinary{}
1428
1429 return NewCCBinary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001430}
1431
Colin Cross6362e272015-10-29 15:25:03 -07001432func (c *CCBinary) ModifyProperties(ctx CCModuleContext) {
Colin Cross0af4b842015-04-30 16:36:18 -07001433 if ctx.Darwin() {
Colin Cross06a931b2015-10-28 17:23:31 -07001434 c.BinaryProperties.Static_executable = proptools.BoolPtr(false)
Colin Cross0af4b842015-04-30 16:36:18 -07001435 }
Colin Cross06a931b2015-10-28 17:23:31 -07001436 if Bool(c.BinaryProperties.Static_executable) {
Colin Cross18b6dc52015-04-28 13:20:37 -07001437 c.dynamicProperties.VariantIsStaticBinary = true
1438 }
1439}
1440
Colin Cross0676e2d2015-04-24 17:39:18 -07001441func (c *CCBinary) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Crosse11befc2015-04-27 17:49:17 -07001442 flags = c.CCLinked.flags(ctx, flags)
Colin Cross21b9a242015-03-24 14:15:58 -07001443
Colin Cross97ba0732015-03-23 17:50:24 -07001444 flags.CFlags = append(flags.CFlags, "-fpie")
1445
Colin Crossf6566ed2015-03-24 11:13:38 -07001446 if ctx.Device() {
Colin Cross06a931b2015-10-28 17:23:31 -07001447 if Bool(c.BinaryProperties.Static_executable) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001448 // Clang driver needs -static to create static executable.
1449 // However, bionic/linker uses -shared to overwrite.
1450 // Linker for x86 targets does not allow coexistance of -static and -shared,
1451 // so we add -static only if -shared is not used.
1452 if !inList("-shared", flags.LdFlags) {
1453 flags.LdFlags = append(flags.LdFlags, "-static")
1454 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001455
Colin Crossed4cf0b2015-03-26 14:43:45 -07001456 flags.LdFlags = append(flags.LdFlags,
1457 "-nostdlib",
1458 "-Bstatic",
1459 "-Wl,--gc-sections",
1460 )
1461
1462 } else {
1463 linker := "/system/bin/linker"
1464 if flags.Toolchain.Is64Bit() {
1465 linker = "/system/bin/linker64"
1466 }
1467
1468 flags.LdFlags = append(flags.LdFlags,
1469 "-nostdlib",
1470 "-Bdynamic",
1471 fmt.Sprintf("-Wl,-dynamic-linker,%s", linker),
1472 "-Wl,--gc-sections",
1473 "-Wl,-z,nocopyreloc",
1474 )
1475 }
Colin Cross0af4b842015-04-30 16:36:18 -07001476 } else if ctx.Darwin() {
1477 flags.LdFlags = append(flags.LdFlags, "-Wl,-headerpad_max_install_names")
Colin Cross3f40fa42015-01-30 17:27:36 -08001478 }
1479
Colin Cross97ba0732015-03-23 17:50:24 -07001480 return flags
Colin Cross3f40fa42015-01-30 17:27:36 -08001481}
1482
Colin Cross97ba0732015-03-23 17:50:24 -07001483func (c *CCBinary) compileModule(ctx common.AndroidModuleContext,
1484 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001485
Colin Cross06a931b2015-10-28 17:23:31 -07001486 if !Bool(c.BinaryProperties.Static_executable) && inList("libc", c.Properties.Static_libs) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001487 ctx.ModuleErrorf("statically linking libc to dynamic executable, please remove libc\n" +
1488 "from static libs or set static_executable: true")
1489 }
1490
1491 outputFile := filepath.Join(common.ModuleOutDir(ctx), c.getStem(ctx))
Dan Albertc403f7c2015-03-18 14:01:18 -07001492 c.out = outputFile
Colin Crossbfae8852015-03-26 14:44:11 -07001493 if c.BinaryProperties.Prefix_symbols != "" {
1494 afterPrefixSymbols := outputFile
1495 outputFile = outputFile + ".intermediate"
1496 TransformBinaryPrefixSymbols(ctx, c.BinaryProperties.Prefix_symbols, outputFile,
1497 ccFlagsToBuilderFlags(flags), afterPrefixSymbols)
1498 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001499
Colin Crossaee540a2015-07-06 17:48:31 -07001500 var linkerDeps []string
1501
Colin Cross97ba0732015-03-23 17:50:24 -07001502 TransformObjToDynamicBinary(ctx, objFiles, deps.SharedLibs, deps.StaticLibs,
Colin Crossaee540a2015-07-06 17:48:31 -07001503 deps.LateStaticLibs, deps.WholeStaticLibs, linkerDeps, deps.CrtBegin, deps.CrtEnd, true,
Colin Cross77b00fa2015-03-16 16:15:49 -07001504 ccFlagsToBuilderFlags(flags), outputFile)
Dan Albertc403f7c2015-03-18 14:01:18 -07001505}
Colin Cross3f40fa42015-01-30 17:27:36 -08001506
Colin Cross97ba0732015-03-23 17:50:24 -07001507func (c *CCBinary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossd350ecd2015-04-28 13:25:36 -07001508 c.installFile = ctx.InstallFile(filepath.Join("bin", c.Properties.Relative_install_path), c.out)
1509}
1510
1511func (c *CCBinary) HostToolPath() string {
1512 if c.HostOrDevice().Host() {
1513 return c.installFile
1514 }
1515 return ""
Dan Albertc403f7c2015-03-18 14:01:18 -07001516}
1517
Colin Cross6002e052015-09-16 16:00:08 -07001518func (c *CCBinary) testPerSrc() bool {
Colin Cross06a931b2015-10-28 17:23:31 -07001519 return Bool(c.BinaryProperties.Test_per_src)
Colin Cross6002e052015-09-16 16:00:08 -07001520}
1521
1522func (c *CCBinary) binary() *CCBinary {
1523 return c
1524}
1525
1526type testPerSrc interface {
1527 binary() *CCBinary
1528 testPerSrc() bool
1529}
1530
1531var _ testPerSrc = (*CCBinary)(nil)
1532
Colin Cross6362e272015-10-29 15:25:03 -07001533func testPerSrcMutator(mctx common.AndroidBottomUpMutatorContext) {
Colin Cross6002e052015-09-16 16:00:08 -07001534 if test, ok := mctx.Module().(testPerSrc); ok {
1535 if test.testPerSrc() {
1536 testNames := make([]string, len(test.binary().Properties.Srcs))
1537 for i, src := range test.binary().Properties.Srcs {
1538 testNames[i] = strings.TrimSuffix(filepath.Base(src), filepath.Ext(src))
1539 }
1540 tests := mctx.CreateLocalVariations(testNames...)
1541 for i, src := range test.binary().Properties.Srcs {
1542 tests[i].(testPerSrc).binary().Properties.Srcs = []string{src}
1543 tests[i].(testPerSrc).binary().BinaryProperties.Stem = mctx.ModuleName() + "_" + testNames[i]
1544 }
1545 }
1546 }
Colin Cross7d5136f2015-05-11 13:39:40 -07001547}
1548
Colin Cross9ffb4f52015-04-24 17:48:09 -07001549type CCTest struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001550 CCBinary
Dan Albertc403f7c2015-03-18 14:01:18 -07001551}
1552
Colin Cross9ffb4f52015-04-24 17:48:09 -07001553func (c *CCTest) flags(ctx common.AndroidModuleContext, flags CCFlags) CCFlags {
Colin Cross0676e2d2015-04-24 17:39:18 -07001554 flags = c.CCBinary.flags(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001555
Colin Cross97ba0732015-03-23 17:50:24 -07001556 flags.CFlags = append(flags.CFlags, "-DGTEST_HAS_STD_STRING")
Colin Crossf6566ed2015-03-24 11:13:38 -07001557 if ctx.Host() {
Colin Cross97ba0732015-03-23 17:50:24 -07001558 flags.CFlags = append(flags.CFlags, "-O0", "-g")
Colin Cross28344522015-04-22 13:07:53 -07001559 flags.LdFlags = append(flags.LdFlags, "-lpthread")
Dan Albertc403f7c2015-03-18 14:01:18 -07001560 }
1561
1562 // TODO(danalbert): Make gtest export its dependencies.
Colin Cross28344522015-04-22 13:07:53 -07001563 flags.CFlags = append(flags.CFlags,
1564 "-I"+filepath.Join(ctx.AConfig().SrcDir(), "external/gtest/include"))
Dan Albertc403f7c2015-03-18 14:01:18 -07001565
Colin Cross21b9a242015-03-24 14:15:58 -07001566 return flags
Dan Albertc403f7c2015-03-18 14:01:18 -07001567}
1568
Colin Cross9ffb4f52015-04-24 17:48:09 -07001569func (c *CCTest) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Dan Willemsene6540452015-10-20 15:21:33 -07001570 depNames.StaticLibs = append(depNames.StaticLibs, "libgtest_main", "libgtest")
Colin Crossa8a93d32015-04-28 13:26:49 -07001571 depNames = c.CCBinary.depNames(ctx, depNames)
Colin Cross21b9a242015-03-24 14:15:58 -07001572 return depNames
Dan Albertc403f7c2015-03-18 14:01:18 -07001573}
1574
Colin Cross9ffb4f52015-04-24 17:48:09 -07001575func (c *CCTest) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Colin Crossf6566ed2015-03-24 11:13:38 -07001576 if ctx.Device() {
Colin Crossa8a93d32015-04-28 13:26:49 -07001577 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
Dan Albertc403f7c2015-03-18 14:01:18 -07001578 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001579 c.CCBinary.installModule(ctx, flags)
Dan Albertc403f7c2015-03-18 14:01:18 -07001580 }
1581}
1582
Colin Cross9ffb4f52015-04-24 17:48:09 -07001583func NewCCTest(test *CCTest, module CCModuleType,
1584 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1585
Colin Cross9ffb4f52015-04-24 17:48:09 -07001586 return NewCCBinary(&test.CCBinary, module, hod, props...)
1587}
1588
1589func CCTestFactory() (blueprint.Module, []interface{}) {
1590 module := &CCTest{}
1591
1592 return NewCCTest(module, module, common.HostAndDeviceSupported)
1593}
1594
Colin Cross2ba19d92015-05-07 15:44:20 -07001595type CCBenchmark struct {
1596 CCBinary
1597}
1598
1599func (c *CCBenchmark) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1600 depNames = c.CCBinary.depNames(ctx, depNames)
Dan Willemsenf8e98b02015-09-11 17:41:44 -07001601 depNames.StaticLibs = append(depNames.StaticLibs, "libbenchmark", "libbase")
Colin Cross2ba19d92015-05-07 15:44:20 -07001602 return depNames
1603}
1604
1605func (c *CCBenchmark) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1606 if ctx.Device() {
1607 ctx.InstallFile("../data/nativetest"+ctx.Arch().ArchType.Multilib[3:]+"/"+ctx.ModuleName(), c.out)
1608 } else {
1609 c.CCBinary.installModule(ctx, flags)
1610 }
1611}
1612
1613func NewCCBenchmark(test *CCBenchmark, module CCModuleType,
1614 hod common.HostOrDeviceSupported, props ...interface{}) (blueprint.Module, []interface{}) {
1615
1616 return NewCCBinary(&test.CCBinary, module, hod, props...)
1617}
1618
1619func CCBenchmarkFactory() (blueprint.Module, []interface{}) {
1620 module := &CCBenchmark{}
1621
1622 return NewCCBenchmark(module, module, common.HostAndDeviceSupported)
1623}
1624
Colin Cross3f40fa42015-01-30 17:27:36 -08001625//
1626// Static library
1627//
1628
Colin Cross97ba0732015-03-23 17:50:24 -07001629func CCLibraryStaticFactory() (blueprint.Module, []interface{}) {
1630 module := &CCLibrary{}
1631 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001632
Colin Cross97ba0732015-03-23 17:50:24 -07001633 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001634}
1635
1636//
1637// Shared libraries
1638//
1639
Colin Cross97ba0732015-03-23 17:50:24 -07001640func CCLibrarySharedFactory() (blueprint.Module, []interface{}) {
1641 module := &CCLibrary{}
1642 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001643
Colin Cross97ba0732015-03-23 17:50:24 -07001644 return NewCCLibrary(module, module, common.HostAndDeviceSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001645}
1646
1647//
1648// Host static library
1649//
1650
Colin Cross97ba0732015-03-23 17:50:24 -07001651func CCLibraryHostStaticFactory() (blueprint.Module, []interface{}) {
1652 module := &CCLibrary{}
1653 module.LibraryProperties.BuildStatic = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001654
Colin Cross97ba0732015-03-23 17:50:24 -07001655 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001656}
1657
1658//
1659// Host Shared libraries
1660//
1661
Colin Cross97ba0732015-03-23 17:50:24 -07001662func CCLibraryHostSharedFactory() (blueprint.Module, []interface{}) {
1663 module := &CCLibrary{}
1664 module.LibraryProperties.BuildShared = true
Colin Cross3f40fa42015-01-30 17:27:36 -08001665
Colin Cross97ba0732015-03-23 17:50:24 -07001666 return NewCCLibrary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001667}
1668
1669//
1670// Host Binaries
1671//
1672
Colin Cross97ba0732015-03-23 17:50:24 -07001673func CCBinaryHostFactory() (blueprint.Module, []interface{}) {
1674 module := &CCBinary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001675
Colin Cross97ba0732015-03-23 17:50:24 -07001676 return NewCCBinary(module, module, common.HostSupported)
Colin Cross3f40fa42015-01-30 17:27:36 -08001677}
1678
1679//
Colin Cross1f8f2342015-03-26 16:09:47 -07001680// Host Tests
1681//
1682
1683func CCTestHostFactory() (blueprint.Module, []interface{}) {
Colin Cross9ffb4f52015-04-24 17:48:09 -07001684 module := &CCTest{}
Colin Cross6002e052015-09-16 16:00:08 -07001685 return NewCCBinary(&module.CCBinary, module, common.HostSupported)
Colin Cross1f8f2342015-03-26 16:09:47 -07001686}
1687
1688//
Colin Cross2ba19d92015-05-07 15:44:20 -07001689// Host Benchmarks
1690//
1691
1692func CCBenchmarkHostFactory() (blueprint.Module, []interface{}) {
1693 module := &CCBenchmark{}
1694 return NewCCBinary(&module.CCBinary, module, common.HostSupported)
1695}
1696
1697//
Colin Crosscfad1192015-11-02 16:43:11 -08001698// Defaults
1699//
1700type CCDefaults struct {
1701 common.AndroidModuleBase
1702 common.DefaultsModule
1703}
1704
1705func (*CCDefaults) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
1706}
1707
1708func CCDefaultsFactory() (blueprint.Module, []interface{}) {
1709 module := &CCDefaults{}
1710
1711 propertyStructs := []interface{}{
1712 &CCBaseProperties{},
1713 &CCLibraryProperties{},
1714 &CCBinaryProperties{},
1715 &CCUnusedProperties{},
1716 }
1717
1718 _, propertyStructs = common.InitAndroidArchModule(module, common.HostOrDeviceSupported(0),
1719 common.Multilib(""), propertyStructs...)
1720
1721 return common.InitDefaultsModule(module, module, propertyStructs...)
1722}
1723
1724//
Colin Cross3f40fa42015-01-30 17:27:36 -08001725// Device libraries shipped with gcc
1726//
1727
1728type toolchainLibrary struct {
Colin Cross97ba0732015-03-23 17:50:24 -07001729 CCLibrary
Colin Cross3f40fa42015-01-30 17:27:36 -08001730}
1731
Colin Cross0676e2d2015-04-24 17:39:18 -07001732func (*toolchainLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Colin Cross3f40fa42015-01-30 17:27:36 -08001733 // toolchain libraries can't have any dependencies
Colin Cross21b9a242015-03-24 14:15:58 -07001734 return CCDeps{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001735}
1736
Colin Cross97ba0732015-03-23 17:50:24 -07001737func ToolchainLibraryFactory() (blueprint.Module, []interface{}) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001738 module := &toolchainLibrary{}
Colin Cross3f40fa42015-01-30 17:27:36 -08001739
Colin Cross97ba0732015-03-23 17:50:24 -07001740 module.LibraryProperties.BuildStatic = true
1741
Colin Crossfa138792015-04-24 17:31:52 -07001742 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth,
Colin Cross21b9a242015-03-24 14:15:58 -07001743 &module.LibraryProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -08001744}
1745
1746func (c *toolchainLibrary) compileModule(ctx common.AndroidModuleContext,
Colin Cross97ba0732015-03-23 17:50:24 -07001747 flags CCFlags, deps CCDeps, objFiles []string) {
Colin Cross3f40fa42015-01-30 17:27:36 -08001748
1749 libName := ctx.ModuleName() + staticLibraryExtension
1750 outputFile := filepath.Join(common.ModuleOutDir(ctx), libName)
1751
1752 CopyGccLib(ctx, libName, ccFlagsToBuilderFlags(flags), outputFile)
1753
1754 c.out = outputFile
1755
1756 ctx.CheckbuildFile(outputFile)
1757}
1758
Colin Cross97ba0732015-03-23 17:50:24 -07001759func (c *toolchainLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc403f7c2015-03-18 14:01:18 -07001760 // Toolchain libraries do not get installed.
1761}
1762
Dan Albertbe961682015-03-18 23:38:50 -07001763// NDK prebuilt libraries.
1764//
1765// These differ from regular prebuilts in that they aren't stripped and usually aren't installed
1766// either (with the exception of the shared STLs, which are installed to the app's directory rather
1767// than to the system image).
1768
1769func getNdkLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, version string) string {
1770 return fmt.Sprintf("%s/prebuilts/ndk/current/platforms/android-%s/arch-%s/usr/lib",
Colin Cross1332b002015-04-07 17:11:30 -07001771 ctx.AConfig().SrcDir(), version, toolchain.Name())
Dan Albertbe961682015-03-18 23:38:50 -07001772}
1773
Dan Albertc3144b12015-04-28 18:17:56 -07001774func ndkPrebuiltModuleToPath(ctx common.AndroidModuleContext, toolchain Toolchain,
1775 ext string, version string) string {
1776
1777 // NDK prebuilts are named like: ndk_NAME.EXT.SDK_VERSION.
1778 // We want to translate to just NAME.EXT
1779 name := strings.Split(strings.TrimPrefix(ctx.ModuleName(), "ndk_"), ".")[0]
1780 dir := getNdkLibDir(ctx, toolchain, version)
1781 return filepath.Join(dir, name+ext)
1782}
1783
1784type ndkPrebuiltObject struct {
1785 ccObject
1786}
1787
Dan Albertc3144b12015-04-28 18:17:56 -07001788func (*ndkPrebuiltObject) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
1789 // NDK objects can't have any dependencies
1790 return CCDeps{}
1791}
1792
1793func NdkPrebuiltObjectFactory() (blueprint.Module, []interface{}) {
1794 module := &ndkPrebuiltObject{}
1795 return newCCBase(&module.CCBase, module, common.DeviceSupported, common.MultilibBoth)
1796}
1797
1798func (c *ndkPrebuiltObject) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1799 deps CCDeps, objFiles []string) {
1800 // A null build step, but it sets up the output path.
1801 if !strings.HasPrefix(ctx.ModuleName(), "ndk_crt") {
1802 ctx.ModuleErrorf("NDK prebuilts must have an ndk_crt prefixed name")
1803 }
1804
1805 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, objectExtension, c.Properties.Sdk_version)
1806}
1807
1808func (c *ndkPrebuiltObject) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
1809 // Objects do not get installed.
1810}
1811
1812var _ ccObjectProvider = (*ndkPrebuiltObject)(nil)
1813
Dan Albertbe961682015-03-18 23:38:50 -07001814type ndkPrebuiltLibrary struct {
1815 CCLibrary
1816}
1817
Colin Cross0676e2d2015-04-24 17:39:18 -07001818func (*ndkPrebuiltLibrary) depNames(ctx common.AndroidBaseContext, depNames CCDeps) CCDeps {
Dan Albertbe961682015-03-18 23:38:50 -07001819 // NDK libraries can't have any dependencies
1820 return CCDeps{}
1821}
1822
1823func NdkPrebuiltLibraryFactory() (blueprint.Module, []interface{}) {
1824 module := &ndkPrebuiltLibrary{}
1825 module.LibraryProperties.BuildShared = true
1826 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1827}
1828
1829func (c *ndkPrebuiltLibrary) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1830 deps CCDeps, objFiles []string) {
1831 // A null build step, but it sets up the output path.
1832 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1833 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1834 }
1835
Colin Crossfa138792015-04-24 17:31:52 -07001836 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001837 c.exportFlags = []string{common.JoinWithPrefix(includeDirs, "-isystem ")}
Dan Albertbe961682015-03-18 23:38:50 -07001838
Dan Albertc3144b12015-04-28 18:17:56 -07001839 c.out = ndkPrebuiltModuleToPath(ctx, flags.Toolchain, sharedLibraryExtension,
1840 c.Properties.Sdk_version)
Dan Albertbe961682015-03-18 23:38:50 -07001841}
1842
1843func (c *ndkPrebuiltLibrary) installModule(ctx common.AndroidModuleContext, flags CCFlags) {
Dan Albertc3144b12015-04-28 18:17:56 -07001844 // NDK prebuilt libraries do not get installed.
Dan Albertbe961682015-03-18 23:38:50 -07001845}
1846
1847// The NDK STLs are slightly different from the prebuilt system libraries:
1848// * Are not specific to each platform version.
1849// * The libraries are not in a predictable location for each STL.
1850
1851type ndkPrebuiltStl struct {
1852 ndkPrebuiltLibrary
1853}
1854
1855type ndkPrebuiltStaticStl struct {
1856 ndkPrebuiltStl
1857}
1858
1859type ndkPrebuiltSharedStl struct {
1860 ndkPrebuiltStl
1861}
1862
1863func NdkPrebuiltSharedStlFactory() (blueprint.Module, []interface{}) {
1864 module := &ndkPrebuiltSharedStl{}
1865 module.LibraryProperties.BuildShared = true
1866 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1867}
1868
1869func NdkPrebuiltStaticStlFactory() (blueprint.Module, []interface{}) {
1870 module := &ndkPrebuiltStaticStl{}
1871 module.LibraryProperties.BuildStatic = true
1872 return NewCCLibrary(&module.CCLibrary, module, common.DeviceSupported)
1873}
1874
1875func getNdkStlLibDir(ctx common.AndroidModuleContext, toolchain Toolchain, stl string) string {
1876 gccVersion := toolchain.GccVersion()
1877 var libDir string
1878 switch stl {
1879 case "libstlport":
1880 libDir = "cxx-stl/stlport/libs"
1881 case "libc++":
1882 libDir = "cxx-stl/llvm-libc++/libs"
1883 case "libgnustl":
1884 libDir = fmt.Sprintf("cxx-stl/gnu-libstdc++/%s/libs", gccVersion)
1885 }
1886
1887 if libDir != "" {
Colin Cross1332b002015-04-07 17:11:30 -07001888 ndkSrcRoot := ctx.AConfig().SrcDir() + "/prebuilts/ndk/current/sources"
Dan Albertbe961682015-03-18 23:38:50 -07001889 return fmt.Sprintf("%s/%s/%s", ndkSrcRoot, libDir, ctx.Arch().Abi)
1890 }
1891
1892 ctx.ModuleErrorf("Unknown NDK STL: %s", stl)
1893 return ""
1894}
1895
1896func (c *ndkPrebuiltStl) compileModule(ctx common.AndroidModuleContext, flags CCFlags,
1897 deps CCDeps, objFiles []string) {
1898 // A null build step, but it sets up the output path.
1899 if !strings.HasPrefix(ctx.ModuleName(), "ndk_lib") {
1900 ctx.ModuleErrorf("NDK prebuilts must have an ndk_lib prefixed name")
1901 }
1902
Colin Crossfa138792015-04-24 17:31:52 -07001903 includeDirs := pathtools.PrefixPaths(c.Properties.Export_include_dirs, common.ModuleSrcDir(ctx))
Colin Cross28344522015-04-22 13:07:53 -07001904 c.exportFlags = []string{includeDirsToFlags(includeDirs)}
Dan Albertbe961682015-03-18 23:38:50 -07001905
1906 libName := strings.TrimPrefix(ctx.ModuleName(), "ndk_")
1907 libExt := sharedLibraryExtension
1908 if c.LibraryProperties.BuildStatic {
1909 libExt = staticLibraryExtension
1910 }
1911
1912 stlName := strings.TrimSuffix(libName, "_shared")
1913 stlName = strings.TrimSuffix(stlName, "_static")
1914 libDir := getNdkStlLibDir(ctx, flags.Toolchain, stlName)
1915 c.out = libDir + "/" + libName + libExt
1916}
1917
Colin Cross6362e272015-10-29 15:25:03 -07001918func linkageMutator(mctx common.AndroidBottomUpMutatorContext) {
Colin Crossed4cf0b2015-03-26 14:43:45 -07001919 if c, ok := mctx.Module().(ccLinkedInterface); ok {
Colin Cross3f40fa42015-01-30 17:27:36 -08001920 var modules []blueprint.Module
Colin Crossed4cf0b2015-03-26 14:43:45 -07001921 if c.buildStatic() && c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001922 modules = mctx.CreateLocalVariations("static", "shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001923 modules[0].(ccLinkedInterface).setStatic(true)
1924 modules[1].(ccLinkedInterface).setStatic(false)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001925 } else if c.buildStatic() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001926 modules = mctx.CreateLocalVariations("static")
Colin Cross18b6dc52015-04-28 13:20:37 -07001927 modules[0].(ccLinkedInterface).setStatic(true)
Colin Crossed4cf0b2015-03-26 14:43:45 -07001928 } else if c.buildShared() {
Colin Cross3f40fa42015-01-30 17:27:36 -08001929 modules = mctx.CreateLocalVariations("shared")
Colin Cross18b6dc52015-04-28 13:20:37 -07001930 modules[0].(ccLinkedInterface).setStatic(false)
Colin Cross3f40fa42015-01-30 17:27:36 -08001931 } else {
Colin Cross97ba0732015-03-23 17:50:24 -07001932 panic(fmt.Errorf("ccLibrary %q not static or shared", mctx.ModuleName()))
Colin Cross3f40fa42015-01-30 17:27:36 -08001933 }
Colin Crossed4cf0b2015-03-26 14:43:45 -07001934
1935 if _, ok := c.(ccLibraryInterface); ok {
1936 reuseFrom := modules[0].(ccLibraryInterface)
1937 for _, m := range modules {
1938 m.(ccLibraryInterface).setReuseFrom(reuseFrom)
Colin Cross3f40fa42015-01-30 17:27:36 -08001939 }
1940 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001941 }
1942}
Colin Cross74d1ec02015-04-28 13:30:13 -07001943
1944// lastUniqueElements returns all unique elements of a slice, keeping the last copy of each
1945// modifies the slice contents in place, and returns a subslice of the original slice
1946func lastUniqueElements(list []string) []string {
1947 totalSkip := 0
1948 for i := len(list) - 1; i >= totalSkip; i-- {
1949 skip := 0
1950 for j := i - 1; j >= totalSkip; j-- {
1951 if list[i] == list[j] {
1952 skip++
1953 } else {
1954 list[j+skip] = list[j]
1955 }
1956 }
1957 totalSkip += skip
1958 }
1959 return list[totalSkip:]
1960}
Colin Cross06a931b2015-10-28 17:23:31 -07001961
1962var Bool = proptools.Bool