Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 1 | // Copyright 2015 Google Inc. All rights reserved. |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 2 | // |
| 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 | |
| 15 | package 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 | |
| 21 | import ( |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 22 | "strconv" |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 23 | "strings" |
| 24 | |
Colin Cross | 97ba073 | 2015-03-23 17:50:24 -0700 | [diff] [blame] | 25 | "github.com/google/blueprint" |
Colin Cross | 06a931b | 2015-10-28 17:23:31 -0700 | [diff] [blame] | 26 | "github.com/google/blueprint/proptools" |
Colin Cross | 97ba073 | 2015-03-23 17:50:24 -0700 | [diff] [blame] | 27 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 28 | "android/soong/android" |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 29 | "android/soong/cc/config" |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 30 | "android/soong/genrule" |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 31 | ) |
| 32 | |
Colin Cross | 463a90e | 2015-06-17 14:20:06 -0700 | [diff] [blame] | 33 | func init() { |
Colin Cross | 798bfce | 2016-10-12 14:28:16 -0700 | [diff] [blame] | 34 | android.RegisterModuleType("cc_defaults", defaultsFactory) |
Colin Cross | 463a90e | 2015-06-17 14:20:06 -0700 | [diff] [blame] | 35 | |
Colin Cross | 1e676be | 2016-10-12 14:38:15 -0700 | [diff] [blame] | 36 | android.PreDepsMutators(func(ctx android.RegisterMutatorsContext) { |
| 37 | ctx.BottomUp("link", linkageMutator).Parallel() |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 38 | ctx.BottomUp("image", vendorMutator).Parallel() |
Colin Cross | 1e676be | 2016-10-12 14:38:15 -0700 | [diff] [blame] | 39 | ctx.BottomUp("ndk_api", ndkApiMutator).Parallel() |
| 40 | ctx.BottomUp("test_per_src", testPerSrcMutator).Parallel() |
| 41 | ctx.BottomUp("begin", beginMutator).Parallel() |
| 42 | }) |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 43 | |
Colin Cross | 1e676be | 2016-10-12 14:38:15 -0700 | [diff] [blame] | 44 | android.PostDepsMutators(func(ctx android.RegisterMutatorsContext) { |
| 45 | ctx.TopDown("asan_deps", sanitizerDepsMutator(asan)) |
| 46 | ctx.BottomUp("asan", sanitizerMutator(asan)).Parallel() |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 47 | |
Colin Cross | 1e676be | 2016-10-12 14:38:15 -0700 | [diff] [blame] | 48 | ctx.TopDown("tsan_deps", sanitizerDepsMutator(tsan)) |
| 49 | ctx.BottomUp("tsan", sanitizerMutator(tsan)).Parallel() |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 50 | |
| 51 | ctx.BottomUp("coverage", coverageLinkingMutator).Parallel() |
Colin Cross | 1e676be | 2016-10-12 14:38:15 -0700 | [diff] [blame] | 52 | }) |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 53 | |
| 54 | pctx.Import("android/soong/cc/config") |
Colin Cross | 463a90e | 2015-06-17 14:20:06 -0700 | [diff] [blame] | 55 | } |
| 56 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 57 | type Deps struct { |
| 58 | SharedLibs, LateSharedLibs []string |
| 59 | StaticLibs, LateStaticLibs, WholeStaticLibs []string |
Colin Cross | 5950f38 | 2016-12-13 12:50:57 -0800 | [diff] [blame] | 60 | HeaderLibs []string |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 61 | |
Colin Cross | 5950f38 | 2016-12-13 12:50:57 -0800 | [diff] [blame] | 62 | ReexportSharedLibHeaders, ReexportStaticLibHeaders, ReexportHeaderLibHeaders []string |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 63 | |
Colin Cross | 8141347 | 2016-04-11 14:37:39 -0700 | [diff] [blame] | 64 | ObjFiles []string |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 65 | |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 66 | GeneratedSources []string |
| 67 | GeneratedHeaders []string |
| 68 | |
Dan Willemsen | b3454ab | 2016-09-28 17:34:58 -0700 | [diff] [blame] | 69 | ReexportGeneratedHeaders []string |
| 70 | |
Colin Cross | 97ba073 | 2015-03-23 17:50:24 -0700 | [diff] [blame] | 71 | CrtBegin, CrtEnd string |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 72 | } |
| 73 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 74 | type PathDeps struct { |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 75 | // Paths to .so files |
| 76 | SharedLibs, LateSharedLibs android.Paths |
| 77 | // Paths to the dependencies to use for .so files (.so.toc files) |
| 78 | SharedLibsDeps, LateSharedLibsDeps android.Paths |
| 79 | // Paths to .a files |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 80 | StaticLibs, LateStaticLibs, WholeStaticLibs android.Paths |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 81 | |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 82 | // Paths to .o files |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 83 | Objs Objects |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 84 | StaticLibObjs Objects |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 85 | WholeStaticLibObjs Objects |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 86 | |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 87 | // Paths to generated source files |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 88 | GeneratedSources android.Paths |
| 89 | GeneratedHeaders android.Paths |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 90 | |
Dan Willemsen | 76f0827 | 2016-07-09 00:14:08 -0700 | [diff] [blame] | 91 | Flags, ReexportedFlags []string |
Dan Willemsen | 847dcc7 | 2016-09-29 12:13:36 -0700 | [diff] [blame] | 92 | ReexportedFlagsDeps android.Paths |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 93 | |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 94 | // Paths to crt*.o files |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 95 | CrtBegin, CrtEnd android.OptionalPath |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 96 | } |
| 97 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 98 | type Flags struct { |
Colin Cross | 2834452 | 2015-04-22 13:07:53 -0700 | [diff] [blame] | 99 | GlobalFlags []string // Flags that apply to C, C++, and assembly source files |
Vishwath Mohan | 83d9f71 | 2017-03-16 11:01:23 -0700 | [diff] [blame] | 100 | ArFlags []string // Flags that apply to ar |
Colin Cross | 2834452 | 2015-04-22 13:07:53 -0700 | [diff] [blame] | 101 | AsFlags []string // Flags that apply to assembly source files |
| 102 | CFlags []string // Flags that apply to C and C++ source files |
| 103 | ConlyFlags []string // Flags that apply to C source files |
| 104 | CppFlags []string // Flags that apply to C++ source files |
| 105 | YaccFlags []string // Flags that apply to Yacc source files |
Colin Cross | 0c461f1 | 2016-10-20 16:11:43 -0700 | [diff] [blame] | 106 | protoFlags []string // Flags that apply to proto source files |
Dan Willemsen | e1240db | 2016-11-03 14:28:51 -0700 | [diff] [blame] | 107 | aidlFlags []string // Flags that apply to aidl source files |
Colin Cross | 2834452 | 2015-04-22 13:07:53 -0700 | [diff] [blame] | 108 | LdFlags []string // Flags that apply to linker command lines |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 109 | libFlags []string // Flags to add libraries early to the link order |
Dan Willemsen | a03cf6d | 2016-09-26 15:45:04 -0700 | [diff] [blame] | 110 | TidyFlags []string // Flags that apply to clang-tidy |
Colin Cross | 91e9004 | 2016-12-02 17:13:24 -0800 | [diff] [blame] | 111 | YasmFlags []string // Flags that apply to yasm assembly source files |
Colin Cross | 2834452 | 2015-04-22 13:07:53 -0700 | [diff] [blame] | 112 | |
Colin Cross | c319948 | 2017-03-30 15:03:04 -0700 | [diff] [blame] | 113 | // Global include flags that apply to C, C++, and assembly source files |
| 114 | // These must be after any module include flags, which will be in GlobalFlags. |
| 115 | SystemIncludeFlags []string |
| 116 | |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 117 | Toolchain config.Toolchain |
Colin Cross | 2834452 | 2015-04-22 13:07:53 -0700 | [diff] [blame] | 118 | Clang bool |
Dan Willemsen | a03cf6d | 2016-09-26 15:45:04 -0700 | [diff] [blame] | 119 | Tidy bool |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 120 | Coverage bool |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 121 | |
| 122 | RequiredInstructionSet string |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 123 | DynamicLinker string |
| 124 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 125 | CFlagsDeps android.Paths // Files depended on by compiler flags |
Colin Cross | 18c0c5a | 2016-12-01 14:45:23 -0800 | [diff] [blame] | 126 | |
| 127 | GroupStaticLibs bool |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 128 | } |
| 129 | |
Colin Cross | 8141347 | 2016-04-11 14:37:39 -0700 | [diff] [blame] | 130 | type ObjectLinkerProperties struct { |
| 131 | // names of other cc_object modules to link into this module using partial linking |
| 132 | Objs []string `android:"arch_variant"` |
| 133 | } |
| 134 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 135 | // Properties used to compile all C or C++ modules |
| 136 | type BaseProperties struct { |
| 137 | // compile module with clang instead of gcc |
| 138 | Clang *bool `android:"arch_variant"` |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 139 | |
| 140 | // Minimum sdk version supported when compiling against the ndk |
| 141 | Sdk_version string |
| 142 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 143 | // don't insert default compiler flags into asflags, cflags, |
| 144 | // cppflags, conlyflags, ldflags, or include_dirs |
| 145 | No_default_compiler_flags *bool |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 146 | |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 147 | // whether this module should be allowed to install onto /vendor as |
| 148 | // well as /system. The two variants will be built separately, one |
| 149 | // like normal, and the other limited to the set of libraries and |
| 150 | // headers that are exposed to /vendor modules. |
| 151 | // |
| 152 | // The vendor variant may be used with a different (newer) /system, |
| 153 | // so it shouldn't have any unversioned runtime dependencies, or |
| 154 | // make assumptions about the system that may not be true in the |
| 155 | // future. |
| 156 | // |
| 157 | // Nothing happens if BOARD_VNDK_VERSION isn't set in the BoardConfig.mk |
| 158 | Vendor_available *bool |
| 159 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 160 | AndroidMkSharedLibs []string `blueprint:"mutated"` |
Colin Cross | bc6fb16 | 2016-05-24 15:39:04 -0700 | [diff] [blame] | 161 | HideFromMake bool `blueprint:"mutated"` |
Colin Cross | b0f2895 | 2016-09-19 16:46:53 -0700 | [diff] [blame] | 162 | PreventInstall bool `blueprint:"mutated"` |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 163 | |
| 164 | UseVndk bool `blueprint:"mutated"` |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 165 | } |
| 166 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 167 | type UnusedProperties struct { |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 168 | Tags []string |
Colin Cross | cfad119 | 2015-11-02 16:43:11 -0800 | [diff] [blame] | 169 | } |
| 170 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 171 | type ModuleContextIntf interface { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 172 | static() bool |
| 173 | staticBinary() bool |
| 174 | clang() bool |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 175 | toolchain() config.Toolchain |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 176 | noDefaultCompilerFlags() bool |
| 177 | sdk() bool |
| 178 | sdkVersion() string |
Dan Willemsen | d2ede87 | 2016-11-18 14:54:24 -0800 | [diff] [blame] | 179 | vndk() bool |
Dan Willemsen | 8146b2f | 2016-03-30 21:00:30 -0700 | [diff] [blame] | 180 | selectedStl() string |
Colin Cross | ce75d2c | 2016-10-06 16:12:58 -0700 | [diff] [blame] | 181 | baseModuleName() string |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 182 | } |
| 183 | |
| 184 | type ModuleContext interface { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 185 | android.ModuleContext |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 186 | ModuleContextIntf |
| 187 | } |
| 188 | |
| 189 | type BaseModuleContext interface { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 190 | android.BaseContext |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 191 | ModuleContextIntf |
| 192 | } |
| 193 | |
Colin Cross | 37047f1 | 2016-12-13 17:06:13 -0800 | [diff] [blame] | 194 | type DepsContext interface { |
| 195 | android.BottomUpMutatorContext |
| 196 | ModuleContextIntf |
| 197 | } |
| 198 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 199 | type feature interface { |
| 200 | begin(ctx BaseModuleContext) |
Colin Cross | 37047f1 | 2016-12-13 17:06:13 -0800 | [diff] [blame] | 201 | deps(ctx DepsContext, deps Deps) Deps |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 202 | flags(ctx ModuleContext, flags Flags) Flags |
| 203 | props() []interface{} |
| 204 | } |
| 205 | |
| 206 | type compiler interface { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 207 | compilerInit(ctx BaseModuleContext) |
Colin Cross | 37047f1 | 2016-12-13 17:06:13 -0800 | [diff] [blame] | 208 | compilerDeps(ctx DepsContext, deps Deps) Deps |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 209 | compilerFlags(ctx ModuleContext, flags Flags) Flags |
| 210 | compilerProps() []interface{} |
| 211 | |
Colin Cross | 76fada0 | 2016-07-27 10:31:13 -0700 | [diff] [blame] | 212 | appendCflags([]string) |
| 213 | appendAsflags([]string) |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 214 | compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 215 | } |
| 216 | |
| 217 | type linker interface { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 218 | linkerInit(ctx BaseModuleContext) |
Colin Cross | 37047f1 | 2016-12-13 17:06:13 -0800 | [diff] [blame] | 219 | linkerDeps(ctx DepsContext, deps Deps) Deps |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 220 | linkerFlags(ctx ModuleContext, flags Flags) Flags |
| 221 | linkerProps() []interface{} |
| 222 | |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 223 | link(ctx ModuleContext, flags Flags, deps PathDeps, objs Objects) android.Path |
Colin Cross | 76fada0 | 2016-07-27 10:31:13 -0700 | [diff] [blame] | 224 | appendLdflags([]string) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 225 | } |
| 226 | |
| 227 | type installer interface { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 228 | installerProps() []interface{} |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 229 | install(ctx ModuleContext, path android.Path) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 230 | inData() bool |
Vishwath Mohan | 1dd8839 | 2017-03-29 22:00:18 -0700 | [diff] [blame] | 231 | inSanitizerDir() bool |
Dan Willemsen | 4aa75ca | 2016-09-28 16:18:03 -0700 | [diff] [blame] | 232 | hostToolPath() android.OptionalPath |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 233 | } |
| 234 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 235 | type dependencyTag struct { |
| 236 | blueprint.BaseDependencyTag |
| 237 | name string |
| 238 | library bool |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 239 | |
| 240 | reexportFlags bool |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 241 | } |
| 242 | |
| 243 | var ( |
Dan Willemsen | b3454ab | 2016-09-28 17:34:58 -0700 | [diff] [blame] | 244 | sharedDepTag = dependencyTag{name: "shared", library: true} |
| 245 | sharedExportDepTag = dependencyTag{name: "shared", library: true, reexportFlags: true} |
| 246 | lateSharedDepTag = dependencyTag{name: "late shared", library: true} |
| 247 | staticDepTag = dependencyTag{name: "static", library: true} |
| 248 | staticExportDepTag = dependencyTag{name: "static", library: true, reexportFlags: true} |
| 249 | lateStaticDepTag = dependencyTag{name: "late static", library: true} |
| 250 | wholeStaticDepTag = dependencyTag{name: "whole static", library: true, reexportFlags: true} |
Colin Cross | 32ec36c | 2016-12-15 07:39:51 -0800 | [diff] [blame] | 251 | headerDepTag = dependencyTag{name: "header", library: true} |
| 252 | headerExportDepTag = dependencyTag{name: "header", library: true, reexportFlags: true} |
Dan Willemsen | b3454ab | 2016-09-28 17:34:58 -0700 | [diff] [blame] | 253 | genSourceDepTag = dependencyTag{name: "gen source"} |
| 254 | genHeaderDepTag = dependencyTag{name: "gen header"} |
| 255 | genHeaderExportDepTag = dependencyTag{name: "gen header", reexportFlags: true} |
| 256 | objDepTag = dependencyTag{name: "obj"} |
| 257 | crtBeginDepTag = dependencyTag{name: "crtbegin"} |
| 258 | crtEndDepTag = dependencyTag{name: "crtend"} |
| 259 | reuseObjTag = dependencyTag{name: "reuse objects"} |
| 260 | ndkStubDepTag = dependencyTag{name: "ndk stub", library: true} |
| 261 | ndkLateStubDepTag = dependencyTag{name: "ndk late stub", library: true} |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 262 | ) |
| 263 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 264 | // Module contains the properties and members used by all C/C++ module types, and implements |
| 265 | // the blueprint.Module interface. It delegates to compiler, linker, and installer interfaces |
| 266 | // to construct the output file. Behavior can be customized with a Customizer interface |
| 267 | type Module struct { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 268 | android.ModuleBase |
| 269 | android.DefaultableModule |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 270 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 271 | Properties BaseProperties |
| 272 | unused UnusedProperties |
Colin Cross | fa13879 | 2015-04-24 17:31:52 -0700 | [diff] [blame] | 273 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 274 | // initialize before calling Init |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 275 | hod android.HostOrDeviceSupported |
| 276 | multilib android.Multilib |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 277 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 278 | // delegates, initialize before calling Init |
Colin Cross | b4ce0ec | 2016-09-13 13:41:39 -0700 | [diff] [blame] | 279 | features []feature |
| 280 | compiler compiler |
| 281 | linker linker |
| 282 | installer installer |
| 283 | stl *stl |
| 284 | sanitize *sanitize |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 285 | coverage *coverage |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 286 | |
| 287 | androidMkSharedLibDeps []string |
Colin Cross | 74d1ec0 | 2015-04-28 13:30:13 -0700 | [diff] [blame] | 288 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 289 | outputFile android.OptionalPath |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 290 | |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 291 | cachedToolchain config.Toolchain |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 292 | |
| 293 | subAndroidMkOnce map[subAndroidMkProvider]bool |
Fabien Sanglard | d61f1f4 | 2017-01-10 16:21:22 -0800 | [diff] [blame] | 294 | |
| 295 | // Flags used to compile this module |
| 296 | flags Flags |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 297 | } |
| 298 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 299 | func (c *Module) Init() (blueprint.Module, []interface{}) { |
| 300 | props := []interface{}{&c.Properties, &c.unused} |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 301 | if c.compiler != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 302 | props = append(props, c.compiler.compilerProps()...) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 303 | } |
| 304 | if c.linker != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 305 | props = append(props, c.linker.linkerProps()...) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 306 | } |
| 307 | if c.installer != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 308 | props = append(props, c.installer.installerProps()...) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 309 | } |
Colin Cross | a8e07cc | 2016-04-04 15:07:06 -0700 | [diff] [blame] | 310 | if c.stl != nil { |
| 311 | props = append(props, c.stl.props()...) |
| 312 | } |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 313 | if c.sanitize != nil { |
| 314 | props = append(props, c.sanitize.props()...) |
| 315 | } |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 316 | if c.coverage != nil { |
| 317 | props = append(props, c.coverage.props()...) |
| 318 | } |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 319 | for _, feature := range c.features { |
| 320 | props = append(props, feature.props()...) |
| 321 | } |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 322 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 323 | _, props = android.InitAndroidArchModule(c, c.hod, c.multilib, props...) |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 324 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 325 | return android.InitDefaultableModule(c, c, props...) |
Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 326 | } |
| 327 | |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 328 | // Returns true for dependency roots (binaries) |
| 329 | // TODO(ccross): also handle dlopenable libraries |
| 330 | func (c *Module) isDependencyRoot() bool { |
| 331 | if root, ok := c.linker.(interface { |
| 332 | isDependencyRoot() bool |
| 333 | }); ok { |
| 334 | return root.isDependencyRoot() |
| 335 | } |
| 336 | return false |
| 337 | } |
| 338 | |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 339 | func (c *Module) vndk() bool { |
| 340 | return c.Properties.UseVndk |
| 341 | } |
| 342 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 343 | type baseModuleContext struct { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 344 | android.BaseContext |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 345 | moduleContextImpl |
| 346 | } |
| 347 | |
Colin Cross | 37047f1 | 2016-12-13 17:06:13 -0800 | [diff] [blame] | 348 | type depsContext struct { |
| 349 | android.BottomUpMutatorContext |
| 350 | moduleContextImpl |
| 351 | } |
| 352 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 353 | type moduleContext struct { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 354 | android.ModuleContext |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 355 | moduleContextImpl |
| 356 | } |
| 357 | |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 358 | // Vendor returns true for vendor modules so that they get installed onto the |
| 359 | // correct partition |
| 360 | func (ctx *moduleContext) Vendor() bool { |
| 361 | return ctx.ModuleContext.Vendor() || ctx.moduleContextImpl.mod.Properties.UseVndk |
| 362 | } |
| 363 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 364 | type moduleContextImpl struct { |
| 365 | mod *Module |
| 366 | ctx BaseModuleContext |
| 367 | } |
| 368 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 369 | func (ctx *moduleContextImpl) clang() bool { |
| 370 | return ctx.mod.clang(ctx.ctx) |
| 371 | } |
| 372 | |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 373 | func (ctx *moduleContextImpl) toolchain() config.Toolchain { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 374 | return ctx.mod.toolchain(ctx.ctx) |
| 375 | } |
| 376 | |
| 377 | func (ctx *moduleContextImpl) static() bool { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 378 | if static, ok := ctx.mod.linker.(interface { |
| 379 | static() bool |
| 380 | }); ok { |
| 381 | return static.static() |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 382 | } |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 383 | return false |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 384 | } |
| 385 | |
| 386 | func (ctx *moduleContextImpl) staticBinary() bool { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 387 | if static, ok := ctx.mod.linker.(interface { |
| 388 | staticBinary() bool |
| 389 | }); ok { |
| 390 | return static.staticBinary() |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 391 | } |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 392 | return false |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 393 | } |
| 394 | |
| 395 | func (ctx *moduleContextImpl) noDefaultCompilerFlags() bool { |
| 396 | return Bool(ctx.mod.Properties.No_default_compiler_flags) |
| 397 | } |
| 398 | |
| 399 | func (ctx *moduleContextImpl) sdk() bool { |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 400 | if ctx.ctx.Device() && !ctx.vndk() { |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 401 | return ctx.mod.Properties.Sdk_version != "" |
| 402 | } |
| 403 | return false |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 404 | } |
| 405 | |
| 406 | func (ctx *moduleContextImpl) sdkVersion() string { |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 407 | if ctx.ctx.Device() { |
Dan Willemsen | 11b2614 | 2017-03-19 18:30:37 -0700 | [diff] [blame] | 408 | if ctx.vndk() { |
| 409 | return "current" |
Dan Willemsen | d2ede87 | 2016-11-18 14:54:24 -0800 | [diff] [blame] | 410 | } else { |
| 411 | return ctx.mod.Properties.Sdk_version |
| 412 | } |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 413 | } |
| 414 | return "" |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 415 | } |
| 416 | |
Dan Willemsen | d2ede87 | 2016-11-18 14:54:24 -0800 | [diff] [blame] | 417 | func (ctx *moduleContextImpl) vndk() bool { |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 418 | return ctx.mod.vndk() |
Dan Willemsen | d2ede87 | 2016-11-18 14:54:24 -0800 | [diff] [blame] | 419 | } |
| 420 | |
Dan Willemsen | 8146b2f | 2016-03-30 21:00:30 -0700 | [diff] [blame] | 421 | func (ctx *moduleContextImpl) selectedStl() string { |
| 422 | if stl := ctx.mod.stl; stl != nil { |
| 423 | return stl.Properties.SelectedStl |
| 424 | } |
| 425 | return "" |
| 426 | } |
| 427 | |
Colin Cross | ce75d2c | 2016-10-06 16:12:58 -0700 | [diff] [blame] | 428 | func (ctx *moduleContextImpl) baseModuleName() string { |
| 429 | return ctx.mod.ModuleBase.BaseModuleName() |
| 430 | } |
| 431 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 432 | func newBaseModule(hod android.HostOrDeviceSupported, multilib android.Multilib) *Module { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 433 | return &Module{ |
| 434 | hod: hod, |
| 435 | multilib: multilib, |
| 436 | } |
| 437 | } |
| 438 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 439 | func newModule(hod android.HostOrDeviceSupported, multilib android.Multilib) *Module { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 440 | module := newBaseModule(hod, multilib) |
Dan Willemsen | a03cf6d | 2016-09-26 15:45:04 -0700 | [diff] [blame] | 441 | module.features = []feature{ |
| 442 | &tidyFeature{}, |
| 443 | } |
Colin Cross | a8e07cc | 2016-04-04 15:07:06 -0700 | [diff] [blame] | 444 | module.stl = &stl{} |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 445 | module.sanitize = &sanitize{} |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 446 | module.coverage = &coverage{} |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 447 | return module |
| 448 | } |
| 449 | |
Colin Cross | ce75d2c | 2016-10-06 16:12:58 -0700 | [diff] [blame] | 450 | func (c *Module) Prebuilt() *android.Prebuilt { |
| 451 | if p, ok := c.linker.(prebuiltLinkerInterface); ok { |
| 452 | return p.prebuilt() |
| 453 | } |
| 454 | return nil |
| 455 | } |
| 456 | |
| 457 | func (c *Module) Name() string { |
| 458 | name := c.ModuleBase.Name() |
Dan Willemsen | 01a9059 | 2017-04-07 15:21:13 -0700 | [diff] [blame] | 459 | if p, ok := c.linker.(interface { |
| 460 | Name(string) string |
| 461 | }); ok { |
Colin Cross | ce75d2c | 2016-10-06 16:12:58 -0700 | [diff] [blame] | 462 | name = p.Name(name) |
| 463 | } |
| 464 | return name |
| 465 | } |
| 466 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 467 | func (c *Module) GenerateAndroidBuildActions(actx android.ModuleContext) { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 468 | ctx := &moduleContext{ |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 469 | ModuleContext: actx, |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 470 | moduleContextImpl: moduleContextImpl{ |
| 471 | mod: c, |
| 472 | }, |
| 473 | } |
| 474 | ctx.ctx = ctx |
| 475 | |
| 476 | flags := Flags{ |
| 477 | Toolchain: c.toolchain(ctx), |
| 478 | Clang: c.clang(ctx), |
| 479 | } |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 480 | if c.compiler != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 481 | flags = c.compiler.compilerFlags(ctx, flags) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 482 | } |
| 483 | if c.linker != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 484 | flags = c.linker.linkerFlags(ctx, flags) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 485 | } |
Colin Cross | a8e07cc | 2016-04-04 15:07:06 -0700 | [diff] [blame] | 486 | if c.stl != nil { |
| 487 | flags = c.stl.flags(ctx, flags) |
| 488 | } |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 489 | if c.sanitize != nil { |
| 490 | flags = c.sanitize.flags(ctx, flags) |
| 491 | } |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 492 | if c.coverage != nil { |
| 493 | flags = c.coverage.flags(ctx, flags) |
| 494 | } |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 495 | for _, feature := range c.features { |
| 496 | flags = feature.flags(ctx, flags) |
| 497 | } |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 498 | if ctx.Failed() { |
| 499 | return |
| 500 | } |
| 501 | |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 502 | flags.CFlags, _ = filterList(flags.CFlags, config.IllegalFlags) |
| 503 | flags.CppFlags, _ = filterList(flags.CppFlags, config.IllegalFlags) |
| 504 | flags.ConlyFlags, _ = filterList(flags.ConlyFlags, config.IllegalFlags) |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 505 | |
Fabien Sanglard | d61f1f4 | 2017-01-10 16:21:22 -0800 | [diff] [blame] | 506 | deps := c.depsToPaths(ctx) |
| 507 | if ctx.Failed() { |
| 508 | return |
| 509 | } |
| 510 | flags.GlobalFlags = append(flags.GlobalFlags, deps.Flags...) |
| 511 | c.flags = flags |
| 512 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 513 | // Optimization to reduce size of build.ninja |
| 514 | // Replace the long list of flags for each file with a module-local variable |
| 515 | ctx.Variable(pctx, "cflags", strings.Join(flags.CFlags, " ")) |
| 516 | ctx.Variable(pctx, "cppflags", strings.Join(flags.CppFlags, " ")) |
| 517 | ctx.Variable(pctx, "asflags", strings.Join(flags.AsFlags, " ")) |
| 518 | flags.CFlags = []string{"$cflags"} |
| 519 | flags.CppFlags = []string{"$cppflags"} |
| 520 | flags.AsFlags = []string{"$asflags"} |
| 521 | |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 522 | var objs Objects |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 523 | if c.compiler != nil { |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 524 | objs = c.compiler.compile(ctx, flags, deps) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 525 | if ctx.Failed() { |
| 526 | return |
| 527 | } |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 528 | } |
| 529 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 530 | if c.linker != nil { |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 531 | outputFile := c.linker.link(ctx, flags, deps, objs) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 532 | if ctx.Failed() { |
| 533 | return |
| 534 | } |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 535 | c.outputFile = android.OptionalPathForPath(outputFile) |
Colin Cross | ce75d2c | 2016-10-06 16:12:58 -0700 | [diff] [blame] | 536 | } |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 537 | |
Colin Cross | ce75d2c | 2016-10-06 16:12:58 -0700 | [diff] [blame] | 538 | if c.installer != nil && !c.Properties.PreventInstall && c.outputFile.Valid() { |
| 539 | c.installer.install(ctx, c.outputFile.Path()) |
| 540 | if ctx.Failed() { |
| 541 | return |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 542 | } |
Dan Albert | c403f7c | 2015-03-18 14:01:18 -0700 | [diff] [blame] | 543 | } |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 544 | } |
| 545 | |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 546 | func (c *Module) toolchain(ctx BaseModuleContext) config.Toolchain { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 547 | if c.cachedToolchain == nil { |
Colin Cross | b98c8b0 | 2016-07-29 13:44:28 -0700 | [diff] [blame] | 548 | c.cachedToolchain = config.FindToolchain(ctx.Os(), ctx.Arch()) |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 549 | } |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 550 | return c.cachedToolchain |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 551 | } |
| 552 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 553 | func (c *Module) begin(ctx BaseModuleContext) { |
| 554 | if c.compiler != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 555 | c.compiler.compilerInit(ctx) |
Colin Cross | 21b9a24 | 2015-03-24 14:15:58 -0700 | [diff] [blame] | 556 | } |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 557 | if c.linker != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 558 | c.linker.linkerInit(ctx) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 559 | } |
Colin Cross | a8e07cc | 2016-04-04 15:07:06 -0700 | [diff] [blame] | 560 | if c.stl != nil { |
| 561 | c.stl.begin(ctx) |
| 562 | } |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 563 | if c.sanitize != nil { |
| 564 | c.sanitize.begin(ctx) |
| 565 | } |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 566 | if c.coverage != nil { |
| 567 | c.coverage.begin(ctx) |
| 568 | } |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 569 | for _, feature := range c.features { |
| 570 | feature.begin(ctx) |
| 571 | } |
Dan Albert | 7fa7b2e | 2016-08-05 16:37:52 -0700 | [diff] [blame] | 572 | if ctx.sdk() { |
| 573 | version, err := normalizeNdkApiLevel(ctx.sdkVersion(), ctx.Arch()) |
| 574 | if err != nil { |
| 575 | ctx.PropertyErrorf("sdk_version", err.Error()) |
| 576 | } |
Dan Albert | 90f7a4d | 2016-11-08 14:34:24 -0800 | [diff] [blame] | 577 | c.Properties.Sdk_version = version |
Dan Albert | 7fa7b2e | 2016-08-05 16:37:52 -0700 | [diff] [blame] | 578 | } |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 579 | } |
| 580 | |
Colin Cross | 37047f1 | 2016-12-13 17:06:13 -0800 | [diff] [blame] | 581 | func (c *Module) deps(ctx DepsContext) Deps { |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 582 | deps := Deps{} |
| 583 | |
| 584 | if c.compiler != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 585 | deps = c.compiler.compilerDeps(ctx, deps) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 586 | } |
| 587 | if c.linker != nil { |
Colin Cross | 42742b8 | 2016-08-01 13:20:05 -0700 | [diff] [blame] | 588 | deps = c.linker.linkerDeps(ctx, deps) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 589 | } |
Colin Cross | a8e07cc | 2016-04-04 15:07:06 -0700 | [diff] [blame] | 590 | if c.stl != nil { |
| 591 | deps = c.stl.deps(ctx, deps) |
| 592 | } |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 593 | if c.sanitize != nil { |
| 594 | deps = c.sanitize.deps(ctx, deps) |
| 595 | } |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 596 | if c.coverage != nil { |
| 597 | deps = c.coverage.deps(ctx, deps) |
| 598 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 599 | for _, feature := range c.features { |
| 600 | deps = feature.deps(ctx, deps) |
| 601 | } |
| 602 | |
| 603 | deps.WholeStaticLibs = lastUniqueElements(deps.WholeStaticLibs) |
| 604 | deps.StaticLibs = lastUniqueElements(deps.StaticLibs) |
| 605 | deps.LateStaticLibs = lastUniqueElements(deps.LateStaticLibs) |
| 606 | deps.SharedLibs = lastUniqueElements(deps.SharedLibs) |
| 607 | deps.LateSharedLibs = lastUniqueElements(deps.LateSharedLibs) |
Colin Cross | 5950f38 | 2016-12-13 12:50:57 -0800 | [diff] [blame] | 608 | deps.HeaderLibs = lastUniqueElements(deps.HeaderLibs) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 609 | |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 610 | for _, lib := range deps.ReexportSharedLibHeaders { |
| 611 | if !inList(lib, deps.SharedLibs) { |
| 612 | ctx.PropertyErrorf("export_shared_lib_headers", "Shared library not in shared_libs: '%s'", lib) |
| 613 | } |
| 614 | } |
| 615 | |
| 616 | for _, lib := range deps.ReexportStaticLibHeaders { |
| 617 | if !inList(lib, deps.StaticLibs) { |
| 618 | ctx.PropertyErrorf("export_static_lib_headers", "Static library not in static_libs: '%s'", lib) |
| 619 | } |
| 620 | } |
| 621 | |
Colin Cross | 5950f38 | 2016-12-13 12:50:57 -0800 | [diff] [blame] | 622 | for _, lib := range deps.ReexportHeaderLibHeaders { |
| 623 | if !inList(lib, deps.HeaderLibs) { |
| 624 | ctx.PropertyErrorf("export_header_lib_headers", "Header library not in header_libs: '%s'", lib) |
| 625 | } |
| 626 | } |
| 627 | |
Dan Willemsen | b3454ab | 2016-09-28 17:34:58 -0700 | [diff] [blame] | 628 | for _, gen := range deps.ReexportGeneratedHeaders { |
| 629 | if !inList(gen, deps.GeneratedHeaders) { |
| 630 | ctx.PropertyErrorf("export_generated_headers", "Generated header module not in generated_headers: '%s'", gen) |
| 631 | } |
| 632 | } |
| 633 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 634 | return deps |
| 635 | } |
| 636 | |
Dan Albert | 7e9d295 | 2016-08-04 13:02:36 -0700 | [diff] [blame] | 637 | func (c *Module) beginMutator(actx android.BottomUpMutatorContext) { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 638 | ctx := &baseModuleContext{ |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 639 | BaseContext: actx, |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 640 | moduleContextImpl: moduleContextImpl{ |
| 641 | mod: c, |
| 642 | }, |
| 643 | } |
| 644 | ctx.ctx = ctx |
| 645 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 646 | c.begin(ctx) |
Dan Albert | 7e9d295 | 2016-08-04 13:02:36 -0700 | [diff] [blame] | 647 | } |
| 648 | |
Colin Cross | 1e676be | 2016-10-12 14:38:15 -0700 | [diff] [blame] | 649 | func (c *Module) DepsMutator(actx android.BottomUpMutatorContext) { |
| 650 | if !c.Enabled() { |
| 651 | return |
| 652 | } |
| 653 | |
Colin Cross | 37047f1 | 2016-12-13 17:06:13 -0800 | [diff] [blame] | 654 | ctx := &depsContext{ |
| 655 | BottomUpMutatorContext: actx, |
Dan Albert | 7e9d295 | 2016-08-04 13:02:36 -0700 | [diff] [blame] | 656 | moduleContextImpl: moduleContextImpl{ |
| 657 | mod: c, |
| 658 | }, |
| 659 | } |
| 660 | ctx.ctx = ctx |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 661 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 662 | deps := c.deps(ctx) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 663 | |
Colin Cross | b5bc4b4 | 2016-07-11 16:11:59 -0700 | [diff] [blame] | 664 | c.Properties.AndroidMkSharedLibs = append(c.Properties.AndroidMkSharedLibs, deps.SharedLibs...) |
| 665 | c.Properties.AndroidMkSharedLibs = append(c.Properties.AndroidMkSharedLibs, deps.LateSharedLibs...) |
Dan Willemsen | 72d3993 | 2016-07-08 23:23:48 -0700 | [diff] [blame] | 666 | |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 667 | variantNdkLibs := []string{} |
| 668 | variantLateNdkLibs := []string{} |
Dan Willemsen | b916b80 | 2017-03-19 13:44:32 -0700 | [diff] [blame] | 669 | if ctx.Os() == android.Android { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 670 | version := ctx.sdkVersion() |
Dan Willemsen | 72d3993 | 2016-07-08 23:23:48 -0700 | [diff] [blame] | 671 | |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 672 | // Rewrites the names of shared libraries into the names of the NDK |
| 673 | // libraries where appropriate. This returns two slices. |
| 674 | // |
| 675 | // The first is a list of non-variant shared libraries (either rewritten |
| 676 | // NDK libraries to the modules in prebuilts/ndk, or not rewritten |
| 677 | // because they are not NDK libraries). |
| 678 | // |
| 679 | // The second is a list of ndk_library modules. These need to be |
| 680 | // separated because they are a variation dependency and must be added |
| 681 | // in a different manner. |
| 682 | rewriteNdkLibs := func(list []string) ([]string, []string) { |
| 683 | variantLibs := []string{} |
| 684 | nonvariantLibs := []string{} |
| 685 | for _, entry := range list { |
Dan Willemsen | b916b80 | 2017-03-19 13:44:32 -0700 | [diff] [blame] | 686 | if ctx.sdk() && inList(entry, ndkPrebuiltSharedLibraries) { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 687 | if !inList(entry, ndkMigratedLibs) { |
| 688 | nonvariantLibs = append(nonvariantLibs, entry+".ndk."+version) |
| 689 | } else { |
| 690 | variantLibs = append(variantLibs, entry+ndkLibrarySuffix) |
| 691 | } |
Dan Willemsen | b916b80 | 2017-03-19 13:44:32 -0700 | [diff] [blame] | 692 | } else if ctx.vndk() && inList(entry, config.LLndkLibraries()) { |
| 693 | nonvariantLibs = append(nonvariantLibs, entry+llndkLibrarySuffix) |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 694 | } else { |
Dan Willemsen | 7cbf5f8 | 2017-03-28 00:08:30 -0700 | [diff] [blame] | 695 | nonvariantLibs = append(nonvariantLibs, entry) |
Dan Willemsen | 72d3993 | 2016-07-08 23:23:48 -0700 | [diff] [blame] | 696 | } |
| 697 | } |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 698 | return nonvariantLibs, variantLibs |
Dan Willemsen | 72d3993 | 2016-07-08 23:23:48 -0700 | [diff] [blame] | 699 | } |
| 700 | |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 701 | deps.SharedLibs, variantNdkLibs = rewriteNdkLibs(deps.SharedLibs) |
| 702 | deps.LateSharedLibs, variantLateNdkLibs = rewriteNdkLibs(deps.LateSharedLibs) |
Dan Willemsen | 72d3993 | 2016-07-08 23:23:48 -0700 | [diff] [blame] | 703 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 704 | |
Colin Cross | 32ec36c | 2016-12-15 07:39:51 -0800 | [diff] [blame] | 705 | for _, lib := range deps.HeaderLibs { |
| 706 | depTag := headerDepTag |
| 707 | if inList(lib, deps.ReexportHeaderLibHeaders) { |
| 708 | depTag = headerExportDepTag |
| 709 | } |
| 710 | actx.AddVariationDependencies(nil, depTag, lib) |
| 711 | } |
Colin Cross | 5950f38 | 2016-12-13 12:50:57 -0800 | [diff] [blame] | 712 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 713 | actx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, wholeStaticDepTag, |
| 714 | deps.WholeStaticLibs...) |
| 715 | |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 716 | for _, lib := range deps.StaticLibs { |
| 717 | depTag := staticDepTag |
| 718 | if inList(lib, deps.ReexportStaticLibHeaders) { |
| 719 | depTag = staticExportDepTag |
| 720 | } |
Colin Cross | 15a0d46 | 2016-07-14 14:49:58 -0700 | [diff] [blame] | 721 | actx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, depTag, lib) |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 722 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 723 | |
| 724 | actx.AddVariationDependencies([]blueprint.Variation{{"link", "static"}}, lateStaticDepTag, |
| 725 | deps.LateStaticLibs...) |
| 726 | |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 727 | for _, lib := range deps.SharedLibs { |
| 728 | depTag := sharedDepTag |
| 729 | if inList(lib, deps.ReexportSharedLibHeaders) { |
| 730 | depTag = sharedExportDepTag |
| 731 | } |
Colin Cross | 15a0d46 | 2016-07-14 14:49:58 -0700 | [diff] [blame] | 732 | actx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, depTag, lib) |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 733 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 734 | |
| 735 | actx.AddVariationDependencies([]blueprint.Variation{{"link", "shared"}}, lateSharedDepTag, |
| 736 | deps.LateSharedLibs...) |
| 737 | |
Colin Cross | 6886183 | 2016-07-08 10:41:41 -0700 | [diff] [blame] | 738 | actx.AddDependency(c, genSourceDepTag, deps.GeneratedSources...) |
Dan Willemsen | b3454ab | 2016-09-28 17:34:58 -0700 | [diff] [blame] | 739 | |
| 740 | for _, gen := range deps.GeneratedHeaders { |
| 741 | depTag := genHeaderDepTag |
| 742 | if inList(gen, deps.ReexportGeneratedHeaders) { |
| 743 | depTag = genHeaderExportDepTag |
| 744 | } |
| 745 | actx.AddDependency(c, depTag, gen) |
| 746 | } |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 747 | |
Colin Cross | 6886183 | 2016-07-08 10:41:41 -0700 | [diff] [blame] | 748 | actx.AddDependency(c, objDepTag, deps.ObjFiles...) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 749 | |
| 750 | if deps.CrtBegin != "" { |
Colin Cross | 6886183 | 2016-07-08 10:41:41 -0700 | [diff] [blame] | 751 | actx.AddDependency(c, crtBeginDepTag, deps.CrtBegin) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 752 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 753 | if deps.CrtEnd != "" { |
Colin Cross | 6886183 | 2016-07-08 10:41:41 -0700 | [diff] [blame] | 754 | actx.AddDependency(c, crtEndDepTag, deps.CrtEnd) |
Colin Cross | 21b9a24 | 2015-03-24 14:15:58 -0700 | [diff] [blame] | 755 | } |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 756 | |
| 757 | version := ctx.sdkVersion() |
| 758 | actx.AddVariationDependencies([]blueprint.Variation{ |
| 759 | {"ndk_api", version}, {"link", "shared"}}, ndkStubDepTag, variantNdkLibs...) |
| 760 | actx.AddVariationDependencies([]blueprint.Variation{ |
| 761 | {"ndk_api", version}, {"link", "shared"}}, ndkLateStubDepTag, variantLateNdkLibs...) |
Colin Cross | 6362e27 | 2015-10-29 15:25:03 -0700 | [diff] [blame] | 762 | } |
Colin Cross | 21b9a24 | 2015-03-24 14:15:58 -0700 | [diff] [blame] | 763 | |
Dan Albert | 7e9d295 | 2016-08-04 13:02:36 -0700 | [diff] [blame] | 764 | func beginMutator(ctx android.BottomUpMutatorContext) { |
| 765 | if c, ok := ctx.Module().(*Module); ok && c.Enabled() { |
| 766 | c.beginMutator(ctx) |
| 767 | } |
| 768 | } |
| 769 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 770 | func (c *Module) clang(ctx BaseModuleContext) bool { |
| 771 | clang := Bool(c.Properties.Clang) |
| 772 | |
| 773 | if c.Properties.Clang == nil { |
| 774 | if ctx.Host() { |
| 775 | clang = true |
| 776 | } |
| 777 | |
| 778 | if ctx.Device() && ctx.AConfig().DeviceUsesClang() { |
| 779 | clang = true |
| 780 | } |
Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 781 | } |
Colin Cross | 2834452 | 2015-04-22 13:07:53 -0700 | [diff] [blame] | 782 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 783 | if !c.toolchain(ctx).ClangSupported() { |
| 784 | clang = false |
| 785 | } |
| 786 | |
| 787 | return clang |
| 788 | } |
| 789 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 790 | // Convert dependencies to paths. Returns a PathDeps containing paths |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 791 | func (c *Module) depsToPaths(ctx android.ModuleContext) PathDeps { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 792 | var depPaths PathDeps |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 793 | |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 794 | // Whether a module can link to another module, taking into |
| 795 | // account NDK linking. |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 796 | checkLinkType := func(from, to *Module) { |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 797 | if from.Target().Os != android.Android { |
| 798 | // Host code is not restricted |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 799 | return |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 800 | } |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 801 | if from.Properties.UseVndk { |
| 802 | // Vendor code is already limited by the vendor mutator |
| 803 | return |
| 804 | } |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 805 | if from.Properties.Sdk_version == "" { |
| 806 | // Platform code can link to anything |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 807 | return |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 808 | } |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 809 | if _, ok := to.linker.(*toolchainLibraryDecorator); ok { |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 810 | // These are always allowed |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 811 | return |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 812 | } |
| 813 | if _, ok := to.linker.(*ndkPrebuiltLibraryLinker); ok { |
| 814 | // These are allowed, but don't set sdk_version |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 815 | return |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 816 | } |
Dan Willemsen | 3c316bc | 2016-07-07 20:41:36 -0700 | [diff] [blame] | 817 | if _, ok := to.linker.(*ndkPrebuiltStlLinker); ok { |
| 818 | // These are allowed, but don't set sdk_version |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 819 | return |
Dan Willemsen | 3c316bc | 2016-07-07 20:41:36 -0700 | [diff] [blame] | 820 | } |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 821 | if _, ok := to.linker.(*stubDecorator); ok { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 822 | // These aren't real libraries, but are the stub shared libraries that are included in |
| 823 | // the NDK. |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 824 | return |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 825 | } |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 826 | if to.Properties.Sdk_version == "" { |
| 827 | // NDK code linking to platform code is never okay. |
| 828 | ctx.ModuleErrorf("depends on non-NDK-built library %q", |
| 829 | ctx.OtherModuleName(to)) |
| 830 | } |
| 831 | |
| 832 | // All this point we know we have two NDK libraries, but we need to |
| 833 | // check that we're not linking against anything built against a higher |
| 834 | // API level, as it is only valid to link against older or equivalent |
| 835 | // APIs. |
| 836 | |
| 837 | if from.Properties.Sdk_version == "current" { |
| 838 | // Current can link against anything. |
| 839 | return |
| 840 | } else if to.Properties.Sdk_version == "current" { |
| 841 | // Current can't be linked against by anything else. |
| 842 | ctx.ModuleErrorf("links %q built against newer API version %q", |
| 843 | ctx.OtherModuleName(to), "current") |
| 844 | } |
| 845 | |
| 846 | fromApi, err := strconv.Atoi(from.Properties.Sdk_version) |
| 847 | if err != nil { |
| 848 | ctx.PropertyErrorf("sdk_version", |
| 849 | "Invalid sdk_version value (must be int): %q", |
| 850 | from.Properties.Sdk_version) |
| 851 | } |
| 852 | toApi, err := strconv.Atoi(to.Properties.Sdk_version) |
| 853 | if err != nil { |
| 854 | ctx.PropertyErrorf("sdk_version", |
| 855 | "Invalid sdk_version value (must be int): %q", |
| 856 | to.Properties.Sdk_version) |
| 857 | } |
| 858 | |
| 859 | if toApi > fromApi { |
| 860 | ctx.ModuleErrorf("links %q built against newer API version %q", |
| 861 | ctx.OtherModuleName(to), to.Properties.Sdk_version) |
| 862 | } |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 863 | } |
| 864 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 865 | ctx.VisitDirectDeps(func(m blueprint.Module) { |
| 866 | name := ctx.OtherModuleName(m) |
| 867 | tag := ctx.OtherModuleDependencyTag(m) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 868 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 869 | a, _ := m.(android.Module) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 870 | if a == nil { |
| 871 | ctx.ModuleErrorf("module %q not an android module", name) |
| 872 | return |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 873 | } |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 874 | |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 875 | cc, _ := m.(*Module) |
| 876 | if cc == nil { |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 877 | switch tag { |
Colin Cross | 068e0fe | 2016-12-13 15:23:47 -0800 | [diff] [blame] | 878 | case android.DefaultsDepTag, android.SourceDepTag: |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 879 | case genSourceDepTag: |
| 880 | if genRule, ok := m.(genrule.SourceFileGenerator); ok { |
| 881 | depPaths.GeneratedSources = append(depPaths.GeneratedSources, |
| 882 | genRule.GeneratedSourceFiles()...) |
| 883 | } else { |
| 884 | ctx.ModuleErrorf("module %q is not a gensrcs or genrule", name) |
| 885 | } |
Dan Willemsen | b3454ab | 2016-09-28 17:34:58 -0700 | [diff] [blame] | 886 | case genHeaderDepTag, genHeaderExportDepTag: |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 887 | if genRule, ok := m.(genrule.SourceFileGenerator); ok { |
| 888 | depPaths.GeneratedHeaders = append(depPaths.GeneratedHeaders, |
| 889 | genRule.GeneratedSourceFiles()...) |
Colin Cross | 5ed99c6 | 2016-11-22 12:55:55 -0800 | [diff] [blame] | 890 | flags := includeDirsToFlags(genRule.GeneratedHeaderDirs()) |
Dan Willemsen | b3454ab | 2016-09-28 17:34:58 -0700 | [diff] [blame] | 891 | depPaths.Flags = append(depPaths.Flags, flags) |
| 892 | if tag == genHeaderExportDepTag { |
| 893 | depPaths.ReexportedFlags = append(depPaths.ReexportedFlags, flags) |
Dan Willemsen | 847dcc7 | 2016-09-29 12:13:36 -0700 | [diff] [blame] | 894 | depPaths.ReexportedFlagsDeps = append(depPaths.ReexportedFlagsDeps, |
| 895 | genRule.GeneratedSourceFiles()...) |
Dan Willemsen | b3454ab | 2016-09-28 17:34:58 -0700 | [diff] [blame] | 896 | } |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 897 | } else { |
| 898 | ctx.ModuleErrorf("module %q is not a genrule", name) |
| 899 | } |
| 900 | default: |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 901 | ctx.ModuleErrorf("depends on non-cc module %q", name) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 902 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 903 | return |
| 904 | } |
| 905 | |
| 906 | if !a.Enabled() { |
Colin Cross | a8f5e9a | 2016-12-13 12:51:11 -0800 | [diff] [blame] | 907 | if ctx.AConfig().AllowMissingDependencies() { |
| 908 | ctx.AddMissingDependencies([]string{name}) |
| 909 | } else { |
| 910 | ctx.ModuleErrorf("depends on disabled module %q", name) |
| 911 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 912 | return |
| 913 | } |
| 914 | |
Colin Cross | a1ad8d1 | 2016-06-01 17:09:44 -0700 | [diff] [blame] | 915 | if a.Target().Os != ctx.Os() { |
| 916 | ctx.ModuleErrorf("OS mismatch between %q and %q", ctx.ModuleName(), name) |
| 917 | return |
| 918 | } |
| 919 | |
| 920 | if a.Target().Arch.ArchType != ctx.Arch().ArchType { |
| 921 | ctx.ModuleErrorf("Arch mismatch between %q and %q", ctx.ModuleName(), name) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 922 | return |
| 923 | } |
| 924 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 925 | if tag == reuseObjTag { |
Colin Cross | bba9904 | 2016-11-23 15:45:05 -0800 | [diff] [blame] | 926 | if l, ok := cc.compiler.(libraryInterface); ok { |
| 927 | depPaths.Objs = depPaths.Objs.Append(l.reuseObjs()) |
| 928 | return |
| 929 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 930 | } |
| 931 | |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 932 | if t, ok := tag.(dependencyTag); ok && t.library { |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 933 | if i, ok := cc.linker.(exportedFlagsProducer); ok { |
Dan Willemsen | 76f0827 | 2016-07-09 00:14:08 -0700 | [diff] [blame] | 934 | flags := i.exportedFlags() |
Dan Willemsen | 847dcc7 | 2016-09-29 12:13:36 -0700 | [diff] [blame] | 935 | deps := i.exportedFlagsDeps() |
Dan Willemsen | 76f0827 | 2016-07-09 00:14:08 -0700 | [diff] [blame] | 936 | depPaths.Flags = append(depPaths.Flags, flags...) |
Dan Willemsen | 847dcc7 | 2016-09-29 12:13:36 -0700 | [diff] [blame] | 937 | depPaths.GeneratedHeaders = append(depPaths.GeneratedHeaders, deps...) |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 938 | |
| 939 | if t.reexportFlags { |
Dan Willemsen | 76f0827 | 2016-07-09 00:14:08 -0700 | [diff] [blame] | 940 | depPaths.ReexportedFlags = append(depPaths.ReexportedFlags, flags...) |
Dan Willemsen | 847dcc7 | 2016-09-29 12:13:36 -0700 | [diff] [blame] | 941 | depPaths.ReexportedFlagsDeps = append(depPaths.ReexportedFlagsDeps, deps...) |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 942 | } |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 943 | } |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 944 | |
Dan Albert | 9e10cd4 | 2016-08-03 14:12:14 -0700 | [diff] [blame] | 945 | checkLinkType(c, cc) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 946 | } |
| 947 | |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 948 | var ptr *android.Paths |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 949 | var depPtr *android.Paths |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 950 | |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 951 | linkFile := cc.outputFile |
| 952 | depFile := android.OptionalPath{} |
| 953 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 954 | switch tag { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 955 | case ndkStubDepTag, sharedDepTag, sharedExportDepTag: |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 956 | ptr = &depPaths.SharedLibs |
| 957 | depPtr = &depPaths.SharedLibsDeps |
| 958 | depFile = cc.linker.(libraryInterface).toc() |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 959 | case lateSharedDepTag, ndkLateStubDepTag: |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 960 | ptr = &depPaths.LateSharedLibs |
| 961 | depPtr = &depPaths.LateSharedLibsDeps |
| 962 | depFile = cc.linker.(libraryInterface).toc() |
Dan Willemsen | 490a8dc | 2016-06-06 18:22:19 -0700 | [diff] [blame] | 963 | case staticDepTag, staticExportDepTag: |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 964 | ptr = &depPaths.StaticLibs |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 965 | case lateStaticDepTag: |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 966 | ptr = &depPaths.LateStaticLibs |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 967 | case wholeStaticDepTag: |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 968 | ptr = &depPaths.WholeStaticLibs |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 969 | staticLib, ok := cc.linker.(libraryInterface) |
| 970 | if !ok || !staticLib.static() { |
Dan Willemsen | a96ff64 | 2016-06-07 12:34:45 -0700 | [diff] [blame] | 971 | ctx.ModuleErrorf("module %q not a static library", name) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 972 | return |
| 973 | } |
| 974 | |
| 975 | if missingDeps := staticLib.getWholeStaticMissingDeps(); missingDeps != nil { |
| 976 | postfix := " (required by " + ctx.OtherModuleName(m) + ")" |
| 977 | for i := range missingDeps { |
| 978 | missingDeps[i] += postfix |
| 979 | } |
| 980 | ctx.AddMissingDependencies(missingDeps) |
| 981 | } |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 982 | depPaths.WholeStaticLibObjs = depPaths.WholeStaticLibObjs.Append(staticLib.objs()) |
Colin Cross | 5950f38 | 2016-12-13 12:50:57 -0800 | [diff] [blame] | 983 | case headerDepTag: |
| 984 | // Nothing |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 985 | case objDepTag: |
Dan Willemsen | 5cb580f | 2016-09-26 17:33:01 -0700 | [diff] [blame] | 986 | depPaths.Objs.objFiles = append(depPaths.Objs.objFiles, linkFile.Path()) |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 987 | case crtBeginDepTag: |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 988 | depPaths.CrtBegin = linkFile |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 989 | case crtEndDepTag: |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 990 | depPaths.CrtEnd = linkFile |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 991 | } |
| 992 | |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 993 | switch tag { |
| 994 | case staticDepTag, staticExportDepTag, lateStaticDepTag: |
| 995 | staticLib, ok := cc.linker.(libraryInterface) |
| 996 | if !ok || !staticLib.static() { |
| 997 | ctx.ModuleErrorf("module %q not a static library", name) |
| 998 | return |
| 999 | } |
| 1000 | |
| 1001 | // When combining coverage files for shared libraries and executables, coverage files |
| 1002 | // in static libraries act as if they were whole static libraries. |
| 1003 | depPaths.StaticLibObjs.coverageFiles = append(depPaths.StaticLibObjs.coverageFiles, |
| 1004 | staticLib.objs().coverageFiles...) |
| 1005 | } |
| 1006 | |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 1007 | if ptr != nil { |
Colin Cross | ce75d2c | 2016-10-06 16:12:58 -0700 | [diff] [blame] | 1008 | if !linkFile.Valid() { |
| 1009 | ctx.ModuleErrorf("module %q missing output file", name) |
| 1010 | return |
| 1011 | } |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 1012 | *ptr = append(*ptr, linkFile.Path()) |
| 1013 | } |
| 1014 | |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 1015 | if depPtr != nil { |
Colin Cross | 26c34ed | 2016-09-30 17:10:16 -0700 | [diff] [blame] | 1016 | dep := depFile |
| 1017 | if !dep.Valid() { |
| 1018 | dep = linkFile |
| 1019 | } |
| 1020 | *depPtr = append(*depPtr, dep.Path()) |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 1021 | } |
| 1022 | }) |
| 1023 | |
| 1024 | return depPaths |
| 1025 | } |
| 1026 | |
| 1027 | func (c *Module) InstallInData() bool { |
| 1028 | if c.installer == nil { |
| 1029 | return false |
| 1030 | } |
Vishwath Mohan | 1dd8839 | 2017-03-29 22:00:18 -0700 | [diff] [blame] | 1031 | return c.installer.inData() |
| 1032 | } |
| 1033 | |
| 1034 | func (c *Module) InstallInSanitizerDir() bool { |
| 1035 | if c.installer == nil { |
| 1036 | return false |
| 1037 | } |
| 1038 | if c.sanitize != nil && c.sanitize.inSanitizerDir() { |
Colin Cross | 9461040 | 2016-08-29 13:41:32 -0700 | [diff] [blame] | 1039 | return true |
| 1040 | } |
Vishwath Mohan | 1dd8839 | 2017-03-29 22:00:18 -0700 | [diff] [blame] | 1041 | return c.installer.inSanitizerDir() |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 1042 | } |
| 1043 | |
Dan Willemsen | 4aa75ca | 2016-09-28 16:18:03 -0700 | [diff] [blame] | 1044 | func (c *Module) HostToolPath() android.OptionalPath { |
| 1045 | if c.installer == nil { |
| 1046 | return android.OptionalPath{} |
| 1047 | } |
| 1048 | return c.installer.hostToolPath() |
| 1049 | } |
| 1050 | |
Colin Cross | 2ba19d9 | 2015-05-07 15:44:20 -0700 | [diff] [blame] | 1051 | // |
Colin Cross | cfad119 | 2015-11-02 16:43:11 -0800 | [diff] [blame] | 1052 | // Defaults |
| 1053 | // |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 1054 | type Defaults struct { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 1055 | android.ModuleBase |
| 1056 | android.DefaultsModule |
Colin Cross | cfad119 | 2015-11-02 16:43:11 -0800 | [diff] [blame] | 1057 | } |
| 1058 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 1059 | func (*Defaults) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
Colin Cross | cfad119 | 2015-11-02 16:43:11 -0800 | [diff] [blame] | 1060 | } |
| 1061 | |
Colin Cross | 1e676be | 2016-10-12 14:38:15 -0700 | [diff] [blame] | 1062 | func (d *Defaults) DepsMutator(ctx android.BottomUpMutatorContext) { |
| 1063 | } |
| 1064 | |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 1065 | func defaultsFactory() (blueprint.Module, []interface{}) { |
Colin Cross | e1d764e | 2016-08-18 14:18:32 -0700 | [diff] [blame] | 1066 | return DefaultsFactory() |
| 1067 | } |
| 1068 | |
| 1069 | func DefaultsFactory(props ...interface{}) (blueprint.Module, []interface{}) { |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 1070 | module := &Defaults{} |
Colin Cross | cfad119 | 2015-11-02 16:43:11 -0800 | [diff] [blame] | 1071 | |
Colin Cross | e1d764e | 2016-08-18 14:18:32 -0700 | [diff] [blame] | 1072 | props = append(props, |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 1073 | &BaseProperties{}, |
| 1074 | &BaseCompilerProperties{}, |
| 1075 | &BaseLinkerProperties{}, |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 1076 | &LibraryProperties{}, |
Colin Cross | 919281a | 2016-04-05 16:42:05 -0700 | [diff] [blame] | 1077 | &FlagExporterProperties{}, |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 1078 | &BinaryLinkerProperties{}, |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 1079 | &TestProperties{}, |
| 1080 | &TestBinaryProperties{}, |
Colin Cross | ca860ac | 2016-01-04 14:34:37 -0800 | [diff] [blame] | 1081 | &UnusedProperties{}, |
| 1082 | &StlProperties{}, |
Colin Cross | 16b2349 | 2016-01-06 14:41:07 -0800 | [diff] [blame] | 1083 | &SanitizeProperties{}, |
Colin Cross | 665dce9 | 2016-04-28 14:50:03 -0700 | [diff] [blame] | 1084 | &StripProperties{}, |
Dan Willemsen | 7424d61 | 2016-09-01 13:45:39 -0700 | [diff] [blame] | 1085 | &InstallerProperties{}, |
Dan Willemsen | a03cf6d | 2016-09-26 15:45:04 -0700 | [diff] [blame] | 1086 | &TidyProperties{}, |
Dan Willemsen | 581341d | 2017-02-09 16:16:31 -0800 | [diff] [blame] | 1087 | &CoverageProperties{}, |
Colin Cross | e1d764e | 2016-08-18 14:18:32 -0700 | [diff] [blame] | 1088 | ) |
Colin Cross | cfad119 | 2015-11-02 16:43:11 -0800 | [diff] [blame] | 1089 | |
Colin Cross | e1d764e | 2016-08-18 14:18:32 -0700 | [diff] [blame] | 1090 | return android.InitDefaultsModule(module, module, props...) |
Colin Cross | cfad119 | 2015-11-02 16:43:11 -0800 | [diff] [blame] | 1091 | } |
| 1092 | |
Dan Willemsen | 4416e5d | 2017-04-06 12:43:22 -0700 | [diff] [blame^] | 1093 | const ( |
| 1094 | // coreMode is the variant used for framework-private libraries, or |
| 1095 | // SDK libraries. (which framework-private libraries can use) |
| 1096 | coreMode = "core" |
| 1097 | |
| 1098 | // vendorMode is the variant used for /vendor code that compiles |
| 1099 | // against the VNDK. |
| 1100 | vendorMode = "vendor" |
| 1101 | ) |
| 1102 | |
| 1103 | func vendorMutator(mctx android.BottomUpMutatorContext) { |
| 1104 | if mctx.Os() != android.Android { |
| 1105 | return |
| 1106 | } |
| 1107 | |
| 1108 | m, ok := mctx.Module().(*Module) |
| 1109 | if !ok { |
| 1110 | return |
| 1111 | } |
| 1112 | |
| 1113 | // Sanity check |
| 1114 | if Bool(m.Properties.Vendor_available) && mctx.Vendor() { |
| 1115 | mctx.PropertyErrorf("vendor_available", |
| 1116 | "doesn't make sense at the same time as `vendor: true` or `proprietary: true`") |
| 1117 | return |
| 1118 | } |
| 1119 | |
| 1120 | if !mctx.DeviceConfig().CompileVndk() { |
| 1121 | // If the device isn't compiling against the VNDK, we always |
| 1122 | // use the core mode. |
| 1123 | mctx.CreateVariations(coreMode) |
| 1124 | } else if _, ok := m.linker.(*llndkStubDecorator); ok { |
| 1125 | // LL-NDK stubs only exist in the vendor variant, since the |
| 1126 | // real libraries will be used in the core variant. |
| 1127 | mctx.CreateVariations(vendorMode) |
| 1128 | } else if Bool(m.Properties.Vendor_available) { |
| 1129 | // This will be available in both /system and /vendor |
| 1130 | mod := mctx.CreateVariations(coreMode, vendorMode) |
| 1131 | mod[1].(*Module).Properties.UseVndk = true |
| 1132 | } else if mctx.Vendor() && m.Properties.Sdk_version == "" { |
| 1133 | // This will be available in /vendor only |
| 1134 | mod := mctx.CreateVariations(vendorMode) |
| 1135 | mod[0].(*Module).Properties.UseVndk = true |
| 1136 | } else { |
| 1137 | // This is either in /system (or similar: /data), or is a |
| 1138 | // modules built with the NDK. Modules built with the NDK |
| 1139 | // will be restricted using the existing link type checks. |
| 1140 | mctx.CreateVariations(coreMode) |
| 1141 | } |
| 1142 | } |
| 1143 | |
Colin Cross | 74d1ec0 | 2015-04-28 13:30:13 -0700 | [diff] [blame] | 1144 | // lastUniqueElements returns all unique elements of a slice, keeping the last copy of each |
| 1145 | // modifies the slice contents in place, and returns a subslice of the original slice |
| 1146 | func lastUniqueElements(list []string) []string { |
| 1147 | totalSkip := 0 |
| 1148 | for i := len(list) - 1; i >= totalSkip; i-- { |
| 1149 | skip := 0 |
| 1150 | for j := i - 1; j >= totalSkip; j-- { |
| 1151 | if list[i] == list[j] { |
| 1152 | skip++ |
| 1153 | } else { |
| 1154 | list[j+skip] = list[j] |
| 1155 | } |
| 1156 | } |
| 1157 | totalSkip += skip |
| 1158 | } |
| 1159 | return list[totalSkip:] |
| 1160 | } |
Colin Cross | 06a931b | 2015-10-28 17:23:31 -0700 | [diff] [blame] | 1161 | |
| 1162 | var Bool = proptools.Bool |