blob: 47676fd9eee6dad66554535f002a1f9b9f0df8cc [file] [log] [blame]
Colin Crossfabb6082018-02-20 17:22:23 -08001// Copyright 2018 Google Inc. All rights reserved.
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
15package java
16
17import (
18 "android/soong/android"
Colin Crossa97c5d32018-03-28 14:58:31 -070019 "strings"
Colin Crossfabb6082018-02-20 17:22:23 -080020
21 "github.com/google/blueprint"
Colin Crossa97c5d32018-03-28 14:58:31 -070022 "github.com/google/blueprint/proptools"
Colin Crossfabb6082018-02-20 17:22:23 -080023)
24
Colin Crossa97c5d32018-03-28 14:58:31 -070025type AndroidLibraryDependency interface {
26 Dependency
27 ExportPackage() android.Path
Colin Cross89c31582018-04-30 15:55:11 -070028 ExportedProguardFlagFiles() android.Paths
Colin Crossa97c5d32018-03-28 14:58:31 -070029}
30
31func init() {
32 android.RegisterModuleType("android_library_import", AARImportFactory)
33 android.RegisterModuleType("android_library", AndroidLibraryFactory)
34}
35
36//
37// AAR (android library)
38//
39
40type androidLibraryProperties struct {
41 BuildAAR bool `blueprint:"mutated"`
42}
43
44type aaptProperties struct {
45 // flags passed to aapt when creating the apk
46 Aaptflags []string
47
48 // list of directories relative to the Blueprints file containing assets.
49 // Defaults to "assets"
50 Asset_dirs []string
51
52 // list of directories relative to the Blueprints file containing
53 // Android resources
54 Resource_dirs []string
55
56 // path to AndroidManifest.xml. If unset, defaults to "AndroidManifest.xml".
57 Manifest *string
58}
59
60type aapt struct {
61 aaptSrcJar android.Path
62 exportPackage android.Path
63 manifestPath android.Path
64 proguardOptionsFile android.Path
65 rroDirs android.Paths
66 rTxt android.Path
67
68 aaptProperties aaptProperties
69}
70
71func (a *aapt) ExportPackage() android.Path {
72 return a.exportPackage
73}
74
75func (a *aapt) aapt2Flags(ctx android.ModuleContext, sdkVersion string) (flags []string, deps android.Paths,
76 resDirs, overlayDirs []globbedResourceDir, overlayFiles, rroDirs android.Paths, manifestPath android.Path) {
77
78 hasVersionCode := false
79 hasVersionName := false
Colin Crossa97c5d32018-03-28 14:58:31 -070080 for _, f := range a.aaptProperties.Aaptflags {
81 if strings.HasPrefix(f, "--version-code") {
82 hasVersionCode = true
83 } else if strings.HasPrefix(f, "--version-name") {
84 hasVersionName = true
Colin Crossa97c5d32018-03-28 14:58:31 -070085 }
86 }
87
88 var linkFlags []string
89
90 // Flags specified in Android.bp
91 linkFlags = append(linkFlags, a.aaptProperties.Aaptflags...)
92
93 linkFlags = append(linkFlags, "--no-static-lib-packages")
94
95 // Find implicit or explicit asset and resource dirs
96 assetDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Asset_dirs, "assets")
97 resourceDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Resource_dirs, "res")
98
99 var linkDeps android.Paths
100
101 // Glob directories into lists of paths
102 for _, dir := range resourceDirs {
103 resDirs = append(resDirs, globbedResourceDir{
104 dir: dir,
105 files: androidResourceGlob(ctx, dir),
106 })
107 resOverlayDirs, resRRODirs := overlayResourceGlob(ctx, dir)
108 overlayDirs = append(overlayDirs, resOverlayDirs...)
109 rroDirs = append(rroDirs, resRRODirs...)
110 }
111
112 var assetFiles android.Paths
113 for _, dir := range assetDirs {
114 assetFiles = append(assetFiles, androidResourceGlob(ctx, dir)...)
115 }
116
117 // App manifest file
118 manifestFile := proptools.StringDefault(a.aaptProperties.Manifest, "AndroidManifest.xml")
119 manifestPath = android.PathForModuleSrc(ctx, manifestFile)
120 linkFlags = append(linkFlags, "--manifest "+manifestPath.String())
121 linkDeps = append(linkDeps, manifestPath)
122
123 linkFlags = append(linkFlags, android.JoinWithPrefix(assetDirs.Strings(), "-A "))
124 linkDeps = append(linkDeps, assetFiles...)
125
126 staticLibs, libDeps, libFlags := aaptLibs(ctx, sdkVersion)
127
128 overlayFiles = append(overlayFiles, staticLibs...)
129 linkDeps = append(linkDeps, libDeps...)
130 linkFlags = append(linkFlags, libFlags...)
131
132 // SDK version flags
133 switch sdkVersion {
134 case "", "current", "system_current", "test_current":
Colin Crossd09b0b62018-04-18 11:06:47 -0700135 sdkVersion = proptools.NinjaEscape([]string{ctx.Config().DefaultAppTargetSdk()})[0]
Colin Crossa97c5d32018-03-28 14:58:31 -0700136 }
137
138 linkFlags = append(linkFlags, "--min-sdk-version "+sdkVersion)
139 linkFlags = append(linkFlags, "--target-sdk-version "+sdkVersion)
140
Colin Crossa97c5d32018-03-28 14:58:31 -0700141 // Version code
142 if !hasVersionCode {
143 linkFlags = append(linkFlags, "--version-code", ctx.Config().PlatformSdkVersion())
144 }
145
146 if !hasVersionName {
Colin Cross402d5e02018-04-25 14:54:06 -0700147 var versionName string
148 if ctx.ModuleName() == "framework-res" {
149 // Some builds set AppsDefaultVersionName() to include the build number ("O-123456"). aapt2 copies the
150 // version name of framework-res into app manifests as compileSdkVersionCodename, which confuses things
151 // if it contains the build number. Use the DefaultAppTargetSdk instead.
152 versionName = ctx.Config().DefaultAppTargetSdk()
153 } else {
154 versionName = ctx.Config().AppsDefaultVersionName()
155 }
156 versionName = proptools.NinjaEscape([]string{versionName})[0]
Colin Crossa97c5d32018-03-28 14:58:31 -0700157 linkFlags = append(linkFlags, "--version-name ", versionName)
158 }
159
160 return linkFlags, linkDeps, resDirs, overlayDirs, overlayFiles, rroDirs, manifestPath
161}
162
163func (a *aapt) deps(ctx android.BottomUpMutatorContext, sdkVersion string) {
164 if !ctx.Config().UnbundledBuild() {
165 sdkDep := decodeSdkDep(ctx, sdkVersion)
166 if sdkDep.frameworkResModule != "" {
167 ctx.AddDependency(ctx.Module(), frameworkResTag, sdkDep.frameworkResModule)
168 }
169 }
170}
171
172func (a *aapt) buildActions(ctx android.ModuleContext, sdkVersion string, extraLinkFlags ...string) {
173 linkFlags, linkDeps, resDirs, overlayDirs, overlayFiles, rroDirs, manifestPath := a.aapt2Flags(ctx, sdkVersion)
174
175 linkFlags = append(linkFlags, extraLinkFlags...)
176
177 packageRes := android.PathForModuleOut(ctx, "package-res.apk")
178 srcJar := android.PathForModuleGen(ctx, "R.jar")
179 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
180 rTxt := android.PathForModuleOut(ctx, "R.txt")
181
182 var compiledRes, compiledOverlay android.Paths
183 for _, dir := range resDirs {
184 compiledRes = append(compiledRes, aapt2Compile(ctx, dir.dir, dir.files).Paths()...)
185 }
186 for _, dir := range overlayDirs {
187 compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir.dir, dir.files).Paths()...)
188 }
189
190 compiledOverlay = append(compiledOverlay, overlayFiles...)
191
192 aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt,
193 linkFlags, linkDeps, compiledRes, compiledOverlay)
194
195 a.aaptSrcJar = srcJar
196 a.exportPackage = packageRes
197 a.manifestPath = manifestPath
198 a.proguardOptionsFile = proguardOptionsFile
199 a.rroDirs = rroDirs
200 a.rTxt = rTxt
201}
202
203// aaptLibs collects libraries from dependencies and sdk_version and converts them into paths
204func aaptLibs(ctx android.ModuleContext, sdkVersion string) (staticLibs, deps android.Paths, flags []string) {
205 var sharedLibs android.Paths
206
207 sdkDep := decodeSdkDep(ctx, sdkVersion)
208 if sdkDep.useFiles {
209 sharedLibs = append(sharedLibs, sdkDep.jar)
210 }
211
212 ctx.VisitDirectDeps(func(module android.Module) {
213 var exportPackage android.Path
214 if aarDep, ok := module.(AndroidLibraryDependency); ok {
215 exportPackage = aarDep.ExportPackage()
216 }
217
218 switch ctx.OtherModuleDependencyTag(module) {
219 case libTag, frameworkResTag:
220 if exportPackage != nil {
221 sharedLibs = append(sharedLibs, exportPackage)
222 }
223 case staticLibTag:
224 if exportPackage != nil {
225 staticLibs = append(staticLibs, exportPackage)
226 }
227 }
228 })
229
230 deps = append(deps, sharedLibs...)
231 deps = append(deps, staticLibs...)
232
233 if len(staticLibs) > 0 {
234 flags = append(flags, "--auto-add-overlay")
235 }
236
237 for _, sharedLib := range sharedLibs {
238 flags = append(flags, "-I "+sharedLib.String())
239 }
240
241 return staticLibs, deps, flags
242}
243
244type AndroidLibrary struct {
245 Library
246 aapt
247
248 androidLibraryProperties androidLibraryProperties
249
250 aarFile android.WritablePath
Colin Cross89c31582018-04-30 15:55:11 -0700251
252 exportedProguardFlagFiles android.Paths
253}
254
255func (a *AndroidLibrary) ExportedProguardFlagFiles() android.Paths {
256 return a.exportedProguardFlagFiles
Colin Crossa97c5d32018-03-28 14:58:31 -0700257}
258
259var _ AndroidLibraryDependency = (*AndroidLibrary)(nil)
260
261func (a *AndroidLibrary) DepsMutator(ctx android.BottomUpMutatorContext) {
262 a.Module.deps(ctx)
263 if !Bool(a.properties.No_framework_libs) && !Bool(a.properties.No_standard_libs) {
264 a.aapt.deps(ctx, String(a.deviceProperties.Sdk_version))
265 }
266}
267
268func (a *AndroidLibrary) GenerateAndroidBuildActions(ctx android.ModuleContext) {
269 a.aapt.buildActions(ctx, String(a.deviceProperties.Sdk_version), "--static-lib")
270
271 ctx.CheckbuildFile(a.proguardOptionsFile)
272 ctx.CheckbuildFile(a.exportPackage)
273 ctx.CheckbuildFile(a.aaptSrcJar)
274
275 // apps manifests are handled by aapt, don't let Module see them
276 a.properties.Manifest = nil
277
278 a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles,
279 a.proguardOptionsFile)
280
281 a.Module.compile(ctx, a.aaptSrcJar)
282
283 a.aarFile = android.PathForOutput(ctx, ctx.ModuleName()+".aar")
284 var res android.Paths
285 if a.androidLibraryProperties.BuildAAR {
286 BuildAAR(ctx, a.aarFile, a.outputFile, a.manifestPath, a.rTxt, res)
287 ctx.CheckbuildFile(a.aarFile)
288 }
Colin Cross89c31582018-04-30 15:55:11 -0700289
290 ctx.VisitDirectDeps(func(m android.Module) {
291 if lib, ok := m.(AndroidLibraryDependency); ok && ctx.OtherModuleDependencyTag(m) == staticLibTag {
292 a.exportedProguardFlagFiles = append(a.exportedProguardFlagFiles, lib.ExportedProguardFlagFiles()...)
293 }
294 })
295
296 a.exportedProguardFlagFiles = android.FirstUniquePaths(a.exportedProguardFlagFiles)
Colin Crossa97c5d32018-03-28 14:58:31 -0700297}
298
299func AndroidLibraryFactory() android.Module {
300 module := &AndroidLibrary{}
301
302 module.AddProperties(
303 &module.Module.properties,
304 &module.Module.deviceProperties,
305 &module.Module.protoProperties,
306 &module.aaptProperties,
307 &module.androidLibraryProperties)
308
309 module.androidLibraryProperties.BuildAAR = true
310
311 android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibCommon)
312 return module
313}
314
Colin Crossfabb6082018-02-20 17:22:23 -0800315//
316// AAR (android library) prebuilts
317//
Colin Crossfabb6082018-02-20 17:22:23 -0800318
319type AARImportProperties struct {
320 Aars []string
321
322 Sdk_version *string
Colin Crossa97c5d32018-03-28 14:58:31 -0700323
324 Static_libs []string
325 Libs []string
Colin Crossfabb6082018-02-20 17:22:23 -0800326}
327
328type AARImport struct {
329 android.ModuleBase
330 prebuilt android.Prebuilt
331
332 properties AARImportProperties
333
334 classpathFile android.WritablePath
335 proguardFlags android.WritablePath
336 exportPackage android.WritablePath
337}
338
Colin Crossa97c5d32018-03-28 14:58:31 -0700339var _ AndroidLibraryDependency = (*AARImport)(nil)
340
341func (a *AARImport) ExportPackage() android.Path {
342 return a.exportPackage
343}
344
Colin Cross89c31582018-04-30 15:55:11 -0700345func (a *AARImport) ExportedProguardFlagFiles() android.Paths {
346 return android.Paths{a.proguardFlags}
347}
348
Colin Crossfabb6082018-02-20 17:22:23 -0800349func (a *AARImport) Prebuilt() *android.Prebuilt {
350 return &a.prebuilt
351}
352
353func (a *AARImport) Name() string {
354 return a.prebuilt.Name(a.ModuleBase.Name())
355}
356
357func (a *AARImport) DepsMutator(ctx android.BottomUpMutatorContext) {
Colin Crossfabb6082018-02-20 17:22:23 -0800358 if !ctx.Config().UnbundledBuild() {
Colin Crossa97c5d32018-03-28 14:58:31 -0700359 sdkDep := decodeSdkDep(ctx, String(a.properties.Sdk_version))
360 if sdkDep.useModule && sdkDep.frameworkResModule != "" {
361 ctx.AddDependency(ctx.Module(), frameworkResTag, sdkDep.frameworkResModule)
Colin Crossfabb6082018-02-20 17:22:23 -0800362 }
363 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700364
365 ctx.AddDependency(ctx.Module(), staticLibTag, a.properties.Libs...)
366 ctx.AddDependency(ctx.Module(), staticLibTag, a.properties.Static_libs...)
Colin Crossfabb6082018-02-20 17:22:23 -0800367}
368
369// Unzip an AAR into its constituent files and directories. Any files in Outputs that don't exist in the AAR will be
370// touched to create an empty file, and any directories in $expectedDirs will be created.
371var unzipAAR = pctx.AndroidStaticRule("unzipAAR",
372 blueprint.RuleParams{
373 Command: `rm -rf $outDir && mkdir -p $outDir $expectedDirs && ` +
374 `unzip -qo -d $outDir $in && touch $out`,
375 },
376 "expectedDirs", "outDir")
377
378func (a *AARImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
379 if len(a.properties.Aars) != 1 {
380 ctx.PropertyErrorf("aars", "exactly one aar is required")
381 return
382 }
383
384 aar := android.PathForModuleSrc(ctx, a.properties.Aars[0])
385
386 extractedAARDir := android.PathForModuleOut(ctx, "aar")
387 extractedResDir := extractedAARDir.Join(ctx, "res")
388 a.classpathFile = extractedAARDir.Join(ctx, "classes.jar")
389 a.proguardFlags = extractedAARDir.Join(ctx, "proguard.txt")
390 manifest := extractedAARDir.Join(ctx, "AndroidManifest.xml")
391
392 ctx.Build(pctx, android.BuildParams{
393 Rule: unzipAAR,
394 Input: aar,
395 Outputs: android.WritablePaths{a.classpathFile, a.proguardFlags, manifest},
396 Description: "unzip AAR",
397 Args: map[string]string{
398 "expectedDirs": extractedResDir.String(),
399 "outDir": extractedAARDir.String(),
400 },
401 })
402
403 compiledResDir := android.PathForModuleOut(ctx, "flat-res")
404 aaptCompileDeps := android.Paths{a.classpathFile}
405 aaptCompileDirs := android.Paths{extractedResDir}
406 flata := compiledResDir.Join(ctx, "gen_res.flata")
407 aapt2CompileDirs(ctx, flata, aaptCompileDirs, aaptCompileDeps)
408
409 a.exportPackage = android.PathForModuleOut(ctx, "package-res.apk")
410 srcJar := android.PathForModuleGen(ctx, "R.jar")
411 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
Colin Crossa97c5d32018-03-28 14:58:31 -0700412 rTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Crossfabb6082018-02-20 17:22:23 -0800413
414 var linkDeps android.Paths
415
416 linkFlags := []string{
417 "--static-lib",
418 "--no-static-lib-packages",
419 "--auto-add-overlay",
420 }
421
422 linkFlags = append(linkFlags, "--manifest "+manifest.String())
423 linkDeps = append(linkDeps, manifest)
424
Colin Crossa97c5d32018-03-28 14:58:31 -0700425 staticLibs, libDeps, libFlags := aaptLibs(ctx, String(a.properties.Sdk_version))
Colin Crossfabb6082018-02-20 17:22:23 -0800426
Colin Crossa97c5d32018-03-28 14:58:31 -0700427 linkDeps = append(linkDeps, libDeps...)
428 linkFlags = append(linkFlags, libFlags...)
Colin Crossfabb6082018-02-20 17:22:23 -0800429
Colin Crossa97c5d32018-03-28 14:58:31 -0700430 overlayRes := append(android.Paths{flata}, staticLibs...)
Colin Crossfabb6082018-02-20 17:22:23 -0800431
Colin Crossa97c5d32018-03-28 14:58:31 -0700432 aapt2Link(ctx, a.exportPackage, srcJar, proguardOptionsFile, rTxt,
433 linkFlags, linkDeps, nil, overlayRes)
Colin Crossfabb6082018-02-20 17:22:23 -0800434}
435
436var _ Dependency = (*AARImport)(nil)
437
438func (a *AARImport) HeaderJars() android.Paths {
439 return android.Paths{a.classpathFile}
440}
441
442func (a *AARImport) ImplementationJars() android.Paths {
443 return android.Paths{a.classpathFile}
444}
445
446func (a *AARImport) AidlIncludeDirs() android.Paths {
447 return nil
448}
449
450var _ android.PrebuiltInterface = (*Import)(nil)
451
452func AARImportFactory() android.Module {
453 module := &AARImport{}
454
455 module.AddProperties(&module.properties)
456
457 android.InitPrebuiltModule(module, &module.properties.Aars)
458 android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibCommon)
459 return module
460}