blob: 956cbab864500cdfff3947df5a4f13f7d5bfb41c [file] [log] [blame]
Colin Cross3bc7ffa2017-11-22 16:19:37 -08001// Copyright 2017 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 (
Colin Crossd09b0b62018-04-18 11:06:47 -070018 "fmt"
Colin Crossa4f08812018-10-02 22:03:40 -070019 "path/filepath"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080020 "reflect"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070021 "regexp"
Colin Crossb69301e2017-12-01 10:48:26 -080022 "sort"
Colin Crossd09b0b62018-04-18 11:06:47 -070023 "strings"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080024 "testing"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070025
26 "github.com/google/blueprint/proptools"
27
28 "android/soong/android"
29 "android/soong/cc"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080030)
31
32var (
33 resourceFiles = []string{
34 "res/layout/layout.xml",
35 "res/values/strings.xml",
36 "res/values-en-rUS/strings.xml",
37 }
38
39 compiledResourceFiles = []string{
40 "aapt2/res/layout_layout.xml.flat",
41 "aapt2/res/values_strings.arsc.flat",
42 "aapt2/res/values-en-rUS_strings.arsc.flat",
43 }
44)
45
Colin Cross98be1bb2019-12-13 20:41:13 -080046func testAppConfig(env map[string]string, bp string, fs map[string][]byte) android.Config {
Colin Cross527012a2017-11-30 22:56:16 -080047 appFS := map[string][]byte{}
48 for k, v := range fs {
49 appFS[k] = v
Colin Cross3bc7ffa2017-11-22 16:19:37 -080050 }
51
Colin Cross527012a2017-11-30 22:56:16 -080052 for _, file := range resourceFiles {
53 appFS[file] = nil
54 }
55
Colin Cross98be1bb2019-12-13 20:41:13 -080056 return testConfig(env, bp, appFS)
Colin Cross527012a2017-11-30 22:56:16 -080057}
58
59func testApp(t *testing.T, bp string) *android.TestContext {
Colin Cross98be1bb2019-12-13 20:41:13 -080060 config := testAppConfig(nil, bp, nil)
Colin Cross527012a2017-11-30 22:56:16 -080061
Colin Cross98be1bb2019-12-13 20:41:13 -080062 ctx := testContext()
Colin Cross527012a2017-11-30 22:56:16 -080063
64 run(t, ctx, config)
65
66 return ctx
Colin Cross3bc7ffa2017-11-22 16:19:37 -080067}
68
69func TestApp(t *testing.T) {
Colin Crossa97c5d32018-03-28 14:58:31 -070070 for _, moduleType := range []string{"android_app", "android_library"} {
71 t.Run(moduleType, func(t *testing.T) {
72 ctx := testApp(t, moduleType+` {
73 name: "foo",
74 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +090075 sdk_version: "current"
Colin Crossa97c5d32018-03-28 14:58:31 -070076 }
77 `)
Colin Cross3bc7ffa2017-11-22 16:19:37 -080078
Colin Crossa97c5d32018-03-28 14:58:31 -070079 foo := ctx.ModuleForTests("foo", "android_common")
Colin Cross3bc7ffa2017-11-22 16:19:37 -080080
Colin Cross31656952018-05-24 16:11:20 -070081 var expectedLinkImplicits []string
82
83 manifestFixer := foo.Output("manifest_fixer/AndroidManifest.xml")
84 expectedLinkImplicits = append(expectedLinkImplicits, manifestFixer.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080085
Colin Crossa97c5d32018-03-28 14:58:31 -070086 frameworkRes := ctx.ModuleForTests("framework-res", "android_common")
87 expectedLinkImplicits = append(expectedLinkImplicits,
88 frameworkRes.Output("package-res.apk").Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080089
Colin Crossa97c5d32018-03-28 14:58:31 -070090 // Test the mapping from input files to compiled output file names
91 compile := foo.Output(compiledResourceFiles[0])
92 if !reflect.DeepEqual(resourceFiles, compile.Inputs.Strings()) {
93 t.Errorf("expected aapt2 compile inputs expected:\n %#v\n got:\n %#v",
94 resourceFiles, compile.Inputs.Strings())
95 }
Colin Crossb69301e2017-12-01 10:48:26 -080096
Colin Crossa97c5d32018-03-28 14:58:31 -070097 compiledResourceOutputs := compile.Outputs.Strings()
98 sort.Strings(compiledResourceOutputs)
Colin Crossb69301e2017-12-01 10:48:26 -080099
Colin Crossa97c5d32018-03-28 14:58:31 -0700100 expectedLinkImplicits = append(expectedLinkImplicits, compiledResourceOutputs...)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800101
Colin Crossa97c5d32018-03-28 14:58:31 -0700102 list := foo.Output("aapt2/res.list")
103 expectedLinkImplicits = append(expectedLinkImplicits, list.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800104
Colin Crossa97c5d32018-03-28 14:58:31 -0700105 // Check that the link rule uses
106 res := ctx.ModuleForTests("foo", "android_common").Output("package-res.apk")
107 if !reflect.DeepEqual(expectedLinkImplicits, res.Implicits.Strings()) {
108 t.Errorf("expected aapt2 link implicits expected:\n %#v\n got:\n %#v",
109 expectedLinkImplicits, res.Implicits.Strings())
110 }
111 })
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800112 }
113}
Colin Cross890ff552017-11-30 20:13:19 -0800114
Colin Crosse560c4a2019-03-19 16:03:11 -0700115func TestAppSplits(t *testing.T) {
116 ctx := testApp(t, `
117 android_app {
118 name: "foo",
119 srcs: ["a.java"],
120 package_splits: ["v4", "v7,hdpi"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900121 sdk_version: "current"
Colin Crosse560c4a2019-03-19 16:03:11 -0700122 }`)
123
124 foo := ctx.ModuleForTests("foo", "android_common")
125
126 expectedOutputs := []string{
127 filepath.Join(buildDir, ".intermediates/foo/android_common/foo.apk"),
128 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v4.apk"),
129 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v7_hdpi.apk"),
130 }
131 for _, expectedOutput := range expectedOutputs {
132 foo.Output(expectedOutput)
133 }
134
Colin Cross41955e82019-05-29 14:40:35 -0700135 outputFiles, err := foo.Module().(*AndroidApp).OutputFiles("")
136 if err != nil {
137 t.Fatal(err)
138 }
139 if g, w := outputFiles.Strings(), expectedOutputs; !reflect.DeepEqual(g, w) {
140 t.Errorf(`want OutputFiles("") = %q, got %q`, w, g)
Colin Crosse560c4a2019-03-19 16:03:11 -0700141 }
142}
143
Sasha Smundak4de27a52020-04-23 09:49:59 -0700144func TestAndroidAppSet(t *testing.T) {
145 ctx, config := testJava(t, `
146 android_app_set {
147 name: "foo",
148 set: "prebuilts/apks/app.apks",
149 prerelease: true,
150 }`)
151 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700152 const packedSplitApks = "foo.zip"
Sasha Smundak4de27a52020-04-23 09:49:59 -0700153 params := module.Output(packedSplitApks)
154 if params.Rule == nil {
155 t.Errorf("expected output %s is missing", packedSplitApks)
156 }
157 if s := params.Args["allow-prereleased"]; s != "true" {
158 t.Errorf("wrong allow-prereleased value: '%s', expected 'true'", s)
159 }
160 mkEntries := android.AndroidMkEntriesForTest(t, config, "", module.Module())[0]
161 actualMaster := mkEntries.EntryMap["LOCAL_APK_SET_MASTER_FILE"]
162 expectedMaster := []string{"foo.apk"}
163 if !reflect.DeepEqual(actualMaster, expectedMaster) {
164 t.Errorf("Unexpected LOCAL_APK_SET_MASTER_FILE value: '%s', expected: '%s',",
165 actualMaster, expectedMaster)
166 }
167}
168
169func TestAndroidAppSet_Variants(t *testing.T) {
170 bp := `
171 android_app_set {
172 name: "foo",
173 set: "prebuilts/apks/app.apks",
174 }`
175 testCases := []struct {
176 name string
177 deviceArch *string
178 deviceSecondaryArch *string
179 aaptPrebuiltDPI []string
180 sdkVersion int
181 expected map[string]string
182 }{
183 {
184 name: "One",
185 deviceArch: proptools.StringPtr("x86"),
186 aaptPrebuiltDPI: []string{"ldpi", "xxhdpi"},
187 sdkVersion: 29,
188 expected: map[string]string{
189 "abis": "X86",
190 "allow-prereleased": "false",
191 "screen-densities": "LDPI,XXHDPI",
192 "sdk-version": "29",
193 "stem": "foo",
194 },
195 },
196 {
197 name: "Two",
198 deviceArch: proptools.StringPtr("x86_64"),
199 deviceSecondaryArch: proptools.StringPtr("x86"),
200 aaptPrebuiltDPI: nil,
201 sdkVersion: 30,
202 expected: map[string]string{
203 "abis": "X86_64,X86",
204 "allow-prereleased": "false",
205 "screen-densities": "all",
206 "sdk-version": "30",
207 "stem": "foo",
208 },
209 },
210 }
211
212 for _, test := range testCases {
213 config := testAppConfig(nil, bp, nil)
214 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
215 config.TestProductVariables.Platform_sdk_version = &test.sdkVersion
216 config.TestProductVariables.DeviceArch = test.deviceArch
217 config.TestProductVariables.DeviceSecondaryArch = test.deviceSecondaryArch
218 ctx := testContext()
219 run(t, ctx, config)
220 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700221 const packedSplitApks = "foo.zip"
Sasha Smundak4de27a52020-04-23 09:49:59 -0700222 params := module.Output(packedSplitApks)
223 for k, v := range test.expected {
224 if actual := params.Args[k]; actual != v {
225 t.Errorf("%s: bad build arg value for '%s': '%s', expected '%s'",
226 test.name, k, actual, v)
227 }
228 }
229 }
230}
231
Jeongik Cha538c0d02019-07-11 15:54:27 +0900232func TestPlatformAPIs(t *testing.T) {
233 testJava(t, `
234 android_app {
235 name: "foo",
236 srcs: ["a.java"],
237 platform_apis: true,
238 }
239 `)
240
241 testJava(t, `
242 android_app {
243 name: "foo",
244 srcs: ["a.java"],
245 sdk_version: "current",
246 }
247 `)
248
249 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
250 android_app {
251 name: "bar",
252 srcs: ["b.java"],
253 }
254 `)
255
256 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
257 android_app {
258 name: "bar",
259 srcs: ["b.java"],
260 sdk_version: "system_current",
261 platform_apis: true,
262 }
263 `)
264}
265
Jeongik Chae403e9e2019-12-07 00:16:24 +0900266func TestAndroidAppLinkType(t *testing.T) {
267 testJava(t, `
268 android_app {
269 name: "foo",
270 srcs: ["a.java"],
271 libs: ["bar"],
272 static_libs: ["baz"],
273 platform_apis: true,
274 }
275
276 java_library {
277 name: "bar",
278 sdk_version: "current",
279 srcs: ["b.java"],
280 }
281
282 android_library {
283 name: "baz",
284 sdk_version: "system_current",
285 srcs: ["c.java"],
286 }
287 `)
288
289 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
290 android_app {
291 name: "foo",
292 srcs: ["a.java"],
293 libs: ["bar"],
294 sdk_version: "current",
295 static_libs: ["baz"],
296 }
297
298 java_library {
299 name: "bar",
300 sdk_version: "current",
301 srcs: ["b.java"],
302 }
303
304 android_library {
305 name: "baz",
306 sdk_version: "system_current",
307 srcs: ["c.java"],
308 }
309 `)
310
311 testJava(t, `
312 android_app {
313 name: "foo",
314 srcs: ["a.java"],
315 libs: ["bar"],
316 sdk_version: "system_current",
317 static_libs: ["baz"],
318 }
319
320 java_library {
321 name: "bar",
322 sdk_version: "current",
323 srcs: ["b.java"],
324 }
325
326 android_library {
327 name: "baz",
328 sdk_version: "system_current",
329 srcs: ["c.java"],
330 }
331 `)
332
333 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
334 android_app {
335 name: "foo",
336 srcs: ["a.java"],
337 libs: ["bar"],
338 sdk_version: "system_current",
339 static_libs: ["baz"],
340 }
341
342 java_library {
343 name: "bar",
344 sdk_version: "current",
345 srcs: ["b.java"],
346 }
347
348 android_library {
349 name: "baz",
350 srcs: ["c.java"],
351 }
352 `)
353}
354
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100355func TestUpdatableApps(t *testing.T) {
356 testCases := []struct {
357 name string
358 bp string
359 expectedError string
360 }{
361 {
362 name: "Stable public SDK",
363 bp: `android_app {
364 name: "foo",
365 srcs: ["a.java"],
366 sdk_version: "29",
Artur Satayev11962102020-04-16 13:43:02 +0100367 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100368 updatable: true,
369 }`,
370 },
371 {
372 name: "Stable system SDK",
373 bp: `android_app {
374 name: "foo",
375 srcs: ["a.java"],
376 sdk_version: "system_29",
Artur Satayev11962102020-04-16 13:43:02 +0100377 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100378 updatable: true,
379 }`,
380 },
381 {
382 name: "Current public SDK",
383 bp: `android_app {
384 name: "foo",
385 srcs: ["a.java"],
386 sdk_version: "current",
Artur Satayev11962102020-04-16 13:43:02 +0100387 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100388 updatable: true,
389 }`,
390 },
391 {
392 name: "Current system SDK",
393 bp: `android_app {
394 name: "foo",
395 srcs: ["a.java"],
396 sdk_version: "system_current",
Artur Satayev11962102020-04-16 13:43:02 +0100397 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100398 updatable: true,
399 }`,
400 },
401 {
402 name: "Current module SDK",
403 bp: `android_app {
404 name: "foo",
405 srcs: ["a.java"],
406 sdk_version: "module_current",
Artur Satayev11962102020-04-16 13:43:02 +0100407 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100408 updatable: true,
409 }`,
410 },
411 {
412 name: "Current core SDK",
413 bp: `android_app {
414 name: "foo",
415 srcs: ["a.java"],
416 sdk_version: "core_current",
Artur Satayev11962102020-04-16 13:43:02 +0100417 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100418 updatable: true,
419 }`,
420 },
421 {
422 name: "No Platform APIs",
423 bp: `android_app {
424 name: "foo",
425 srcs: ["a.java"],
426 platform_apis: true,
Artur Satayev11962102020-04-16 13:43:02 +0100427 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100428 updatable: true,
429 }`,
430 expectedError: "Updatable apps must use stable SDKs",
431 },
432 {
433 name: "No Core Platform APIs",
434 bp: `android_app {
435 name: "foo",
436 srcs: ["a.java"],
437 sdk_version: "core_platform",
Artur Satayev11962102020-04-16 13:43:02 +0100438 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100439 updatable: true,
440 }`,
441 expectedError: "Updatable apps must use stable SDKs",
442 },
443 {
444 name: "No unspecified APIs",
445 bp: `android_app {
446 name: "foo",
447 srcs: ["a.java"],
448 updatable: true,
Artur Satayev11962102020-04-16 13:43:02 +0100449 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100450 }`,
451 expectedError: "Updatable apps must use stable SDK",
452 },
Artur Satayev11962102020-04-16 13:43:02 +0100453 {
454 name: "Must specify min_sdk_version",
455 bp: `android_app {
456 name: "app_without_min_sdk_version",
457 srcs: ["a.java"],
458 sdk_version: "29",
459 updatable: true,
460 }`,
461 expectedError: "updatable apps must set min_sdk_version.",
462 },
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100463 }
464
465 for _, test := range testCases {
466 t.Run(test.name, func(t *testing.T) {
467 if test.expectedError == "" {
468 testJava(t, test.bp)
469 } else {
470 testJavaError(t, test.expectedError, test.bp)
471 }
472 })
473 }
474}
475
Jooyung Hanaf7f91f2020-04-29 14:01:06 +0900476func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
477 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
478 android_app {
479 name: "foo",
480 srcs: ["a.java"],
481 updatable: true,
482 sdk_version: "current",
483 min_sdk_version: "current",
484 jni_libs: ["libjni"],
485 }
486
487 cc_library {
488 name: "libjni",
489 stl: "none",
490 system_shared_libs: [],
491 sdk_version: "current",
492 }
493 `)
494}
495
496func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
497 bp := cc.GatherRequiredDepsForTest(android.Android) + `
498 android_app {
499 name: "foo",
500 srcs: ["a.java"],
501 updatable: true,
502 sdk_version: "current",
503 min_sdk_version: "29",
504 jni_libs: ["libjni"],
505 }
506
507 cc_library {
508 name: "libjni",
509 stl: "none",
510 system_shared_libs: [],
511 sdk_version: "29",
512 }
513
514 ndk_prebuilt_object {
515 name: "ndk_crtbegin_so.29",
516 sdk_version: "29",
517 }
518
519 ndk_prebuilt_object {
520 name: "ndk_crtend_so.29",
521 sdk_version: "29",
522 }
523 `
524 fs := map[string][]byte{
525 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
526 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
527 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
528 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
529 }
530
531 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
532
533 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
534 var crtbeginFound, crtendFound bool
535 for _, input := range inputs {
536 switch input.String() {
537 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o":
538 crtbeginFound = true
539 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o":
540 crtendFound = true
541 }
542 }
543 if !crtbeginFound || !crtendFound {
544 t.Error("should link with ndk_crtbegin_so.29 and ndk_crtend_so.29")
545 }
546}
547
548func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
549 bp := cc.GatherRequiredDepsForTest(android.Android) + `
550 android_app {
551 name: "foo",
552 srcs: ["a.java"],
553 updatable: true,
554 sdk_version: "current",
555 min_sdk_version: "29", // this APK should support 29
556 jni_libs: ["libjni"],
557 }
558
559 cc_library {
560 name: "libjni",
561 stl: "none",
562 sdk_version: "current",
563 }
564 `
565 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
566}
567
568func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
569 bp := cc.GatherRequiredDepsForTest(android.Android) + `
570 android_app {
571 name: "foo",
572 srcs: ["a.java"],
573 updatable: true,
574 sdk_version: "current",
575 min_sdk_version: "29", // this APK should support 29
576 jni_libs: ["libjni"],
577 }
578
579 cc_library {
580 name: "libjni",
581 stl: "none",
582 shared_libs: ["libbar"],
583 system_shared_libs: [],
584 sdk_version: "27",
585 }
586
587 cc_library {
588 name: "libbar",
589 stl: "none",
590 system_shared_libs: [],
591 sdk_version: "current",
592 }
593 `
594 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
595}
596
Colin Cross0ddae7f2019-02-07 15:30:01 -0800597func TestResourceDirs(t *testing.T) {
598 testCases := []struct {
599 name string
600 prop string
601 resources []string
602 }{
603 {
604 name: "no resource_dirs",
605 prop: "",
606 resources: []string{"res/res/values/strings.xml"},
607 },
608 {
609 name: "resource_dirs",
610 prop: `resource_dirs: ["res"]`,
611 resources: []string{"res/res/values/strings.xml"},
612 },
613 {
614 name: "empty resource_dirs",
615 prop: `resource_dirs: []`,
616 resources: nil,
617 },
618 }
619
620 fs := map[string][]byte{
621 "res/res/values/strings.xml": nil,
622 }
623
624 bp := `
625 android_app {
626 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900627 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800628 %s
629 }
630 `
631
632 for _, testCase := range testCases {
633 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800634 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
635 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800636 run(t, ctx, config)
637
638 module := ctx.ModuleForTests("foo", "android_common")
639 resourceList := module.MaybeOutput("aapt2/res.list")
640
641 var resources []string
642 if resourceList.Rule != nil {
643 for _, compiledResource := range resourceList.Inputs.Strings() {
644 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
645 }
646 }
647
648 if !reflect.DeepEqual(resources, testCase.resources) {
649 t.Errorf("expected resource files %q, got %q",
650 testCase.resources, resources)
651 }
652 })
653 }
654}
655
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800656func TestLibraryAssets(t *testing.T) {
657 bp := `
658 android_app {
659 name: "foo",
660 sdk_version: "current",
661 static_libs: ["lib1", "lib2", "lib3"],
662 }
663
664 android_library {
665 name: "lib1",
666 sdk_version: "current",
667 asset_dirs: ["assets_a"],
668 }
669
670 android_library {
671 name: "lib2",
672 sdk_version: "current",
673 }
674
675 android_library {
676 name: "lib3",
677 sdk_version: "current",
678 static_libs: ["lib4"],
679 }
680
681 android_library {
682 name: "lib4",
683 sdk_version: "current",
684 asset_dirs: ["assets_b"],
685 }
686 `
687
688 testCases := []struct {
689 name string
690 assetFlag string
691 assetPackages []string
692 }{
693 {
694 name: "foo",
695 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
696 assetPackages: []string{
697 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
698 buildDir + "/.intermediates/lib1/android_common/assets.zip",
699 buildDir + "/.intermediates/lib3/android_common/assets.zip",
700 },
701 },
702 {
703 name: "lib1",
704 assetFlag: "-A assets_a",
705 },
706 {
707 name: "lib2",
708 },
709 {
710 name: "lib3",
711 assetPackages: []string{
712 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
713 buildDir + "/.intermediates/lib4/android_common/assets.zip",
714 },
715 },
716 {
717 name: "lib4",
718 assetFlag: "-A assets_b",
719 },
720 }
721 ctx := testApp(t, bp)
722
723 for _, test := range testCases {
724 t.Run(test.name, func(t *testing.T) {
725 m := ctx.ModuleForTests(test.name, "android_common")
726
727 // Check asset flag in aapt2 link flags
728 var aapt2link android.TestingBuildParams
729 if len(test.assetPackages) > 0 {
730 aapt2link = m.Output("aapt2/package-res.apk")
731 } else {
732 aapt2link = m.Output("package-res.apk")
733 }
734 aapt2Flags := aapt2link.Args["flags"]
735 if test.assetFlag != "" {
736 if !strings.Contains(aapt2Flags, test.assetFlag) {
737 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
738 }
739 } else {
740 if strings.Contains(aapt2Flags, " -A ") {
741 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
742 }
743 }
744
745 // Check asset merge rule.
746 if len(test.assetPackages) > 0 {
747 mergeAssets := m.Output("package-res.apk")
748 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
749 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
750 mergeAssets.Inputs.Strings(), test.assetPackages)
751 }
752 }
753 })
754 }
755}
756
Colin Crossbec85302019-02-13 13:15:46 -0800757func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800758 testCases := []struct {
759 name string
760 enforceRROTargets []string
761 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800762 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800763 overlayFiles map[string][]string
764 rroDirs map[string][]string
765 }{
766 {
767 name: "no RRO",
768 enforceRROTargets: nil,
769 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800770 resourceFiles: map[string][]string{
771 "foo": nil,
772 "bar": {"bar/res/res/values/strings.xml"},
773 "lib": nil,
774 "lib2": {"lib2/res/res/values/strings.xml"},
775 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800776 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800777 "foo": {
778 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800779 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000780 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800781 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800782 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
783 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000784 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800785 },
Colin Crossbec85302019-02-13 13:15:46 -0800786 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800787 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
788 "device/vendor/blah/overlay/bar/res/values/strings.xml",
789 },
Colin Crossbec85302019-02-13 13:15:46 -0800790 "lib": {
791 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
792 "lib/res/res/values/strings.xml",
793 "device/vendor/blah/overlay/lib/res/values/strings.xml",
794 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800795 },
796 rroDirs: map[string][]string{
797 "foo": nil,
798 "bar": nil,
799 },
800 },
801 {
802 name: "enforce RRO on foo",
803 enforceRROTargets: []string{"foo"},
804 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800805 resourceFiles: map[string][]string{
806 "foo": nil,
807 "bar": {"bar/res/res/values/strings.xml"},
808 "lib": nil,
809 "lib2": {"lib2/res/res/values/strings.xml"},
810 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800811 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800812 "foo": {
813 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800814 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000815 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800816 "foo/res/res/values/strings.xml",
817 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
818 },
Colin Crossbec85302019-02-13 13:15:46 -0800819 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800820 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
821 "device/vendor/blah/overlay/bar/res/values/strings.xml",
822 },
Colin Crossbec85302019-02-13 13:15:46 -0800823 "lib": {
824 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
825 "lib/res/res/values/strings.xml",
826 "device/vendor/blah/overlay/lib/res/values/strings.xml",
827 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800828 },
Colin Crossc1c37552019-01-31 11:42:41 -0800829
Colin Cross5c4791c2019-02-01 11:44:44 -0800830 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800831 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000832 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800833 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
834 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000835 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800836 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800837 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800838 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800839 },
840 },
841 {
842 name: "enforce RRO on all",
843 enforceRROTargets: []string{"*"},
844 enforceRROExcludedOverlays: []string{
845 // Excluding specific apps/res directories also allowed.
846 "device/vendor/blah/static_overlay/foo",
847 "device/vendor/blah/static_overlay/bar/res",
848 },
Colin Crossbec85302019-02-13 13:15:46 -0800849 resourceFiles: map[string][]string{
850 "foo": nil,
851 "bar": {"bar/res/res/values/strings.xml"},
852 "lib": nil,
853 "lib2": {"lib2/res/res/values/strings.xml"},
854 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800855 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800856 "foo": {
857 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800858 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000859 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800860 "foo/res/res/values/strings.xml",
861 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
862 },
Colin Crossbec85302019-02-13 13:15:46 -0800863 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
864 "lib": {
865 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
866 "lib/res/res/values/strings.xml",
867 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800868 },
869 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800870 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000871 "device:device/vendor/blah/overlay/foo/res",
872 "product:product/vendor/blah/overlay/foo/res",
873 // Lib dep comes after the direct deps
874 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800875 },
Anton Hansson53c88442019-03-18 15:53:16 +0000876 "bar": {"device:device/vendor/blah/overlay/bar/res"},
877 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800878 },
879 },
880 }
881
Anton Hansson53c88442019-03-18 15:53:16 +0000882 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800883 "device/vendor/blah/overlay",
884 "device/vendor/blah/overlay2",
885 "device/vendor/blah/static_overlay",
886 }
887
Anton Hansson53c88442019-03-18 15:53:16 +0000888 productResourceOverlays := []string{
889 "product/vendor/blah/overlay",
890 }
891
Colin Cross890ff552017-11-30 20:13:19 -0800892 fs := map[string][]byte{
893 "foo/res/res/values/strings.xml": nil,
894 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800895 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800896 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800897 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
898 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800899 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800900 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
901 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
902 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000903 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800904 }
905
906 bp := `
907 android_app {
908 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900909 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800910 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000911 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800912 }
913
914 android_app {
915 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900916 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800917 resource_dirs: ["bar/res"],
918 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800919
920 android_library {
921 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900922 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800923 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800924 static_libs: ["lib2"],
925 }
926
927 android_library {
928 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900929 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800930 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800931 }
Anton Hansson53c88442019-03-18 15:53:16 +0000932
933 // This library has the same resources as lib (should not lead to dupe RROs)
934 android_library {
935 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900936 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000937 resource_dirs: ["lib/res"]
938 }
Colin Cross890ff552017-11-30 20:13:19 -0800939 `
940
Colin Cross5c4791c2019-02-01 11:44:44 -0800941 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800942 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800943 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000944 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
945 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800946 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800947 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800948 }
949 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800950 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800951 }
952
Colin Cross98be1bb2019-12-13 20:41:13 -0800953 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800954 run(t, ctx, config)
955
Colin Crossbec85302019-02-13 13:15:46 -0800956 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
957 for _, o := range list {
958 res := module.MaybeOutput(o)
959 if res.Rule != nil {
960 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
961 // verify the inputs to the .arsc.flat rule.
962 files = append(files, res.Inputs.Strings()...)
963 } else {
964 // Otherwise, verify the full path to the output of the other module
965 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000966 }
Colin Cross890ff552017-11-30 20:13:19 -0800967 }
Colin Crossbec85302019-02-13 13:15:46 -0800968 return files
Colin Cross890ff552017-11-30 20:13:19 -0800969 }
970
Colin Crossbec85302019-02-13 13:15:46 -0800971 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
972 module := ctx.ModuleForTests(moduleName, "android_common")
973 resourceList := module.MaybeOutput("aapt2/res.list")
974 if resourceList.Rule != nil {
975 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000976 }
Colin Crossbec85302019-02-13 13:15:46 -0800977 overlayList := module.MaybeOutput("aapt2/overlay.list")
978 if overlayList.Rule != nil {
979 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
980 }
981
Anton Hansson53c88442019-03-18 15:53:16 +0000982 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
983 var prefix string
984 if d.overlayType == device {
985 prefix = "device:"
986 } else if d.overlayType == product {
987 prefix = "product:"
988 } else {
989 t.Fatalf("Unexpected overlayType %d", d.overlayType)
990 }
991 rroDirs = append(rroDirs, prefix+d.path.String())
992 }
Colin Crossbec85302019-02-13 13:15:46 -0800993
994 return resourceFiles, overlayFiles, rroDirs
995 }
996
997 modules := []string{"foo", "bar", "lib", "lib2"}
998 for _, module := range modules {
999 resourceFiles, overlayFiles, rroDirs := getResources(module)
1000
1001 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
1002 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
1003 module, testCase.resourceFiles[module], resourceFiles)
1004 }
1005 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
1006 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
1007 module, testCase.overlayFiles[module], overlayFiles)
1008 }
1009 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +00001010 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -08001011 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +00001012 }
Colin Cross890ff552017-11-30 20:13:19 -08001013 }
Colin Cross890ff552017-11-30 20:13:19 -08001014 })
1015 }
1016}
Colin Crossd09b0b62018-04-18 11:06:47 -07001017
1018func TestAppSdkVersion(t *testing.T) {
1019 testCases := []struct {
1020 name string
1021 sdkVersion string
1022 platformSdkInt int
1023 platformSdkCodename string
1024 platformSdkFinal bool
1025 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +09001026 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -07001027 }{
1028 {
1029 name: "current final SDK",
1030 sdkVersion: "current",
1031 platformSdkInt: 27,
1032 platformSdkCodename: "REL",
1033 platformSdkFinal: true,
1034 expectedMinSdkVersion: "27",
1035 },
1036 {
1037 name: "current non-final SDK",
1038 sdkVersion: "current",
1039 platformSdkInt: 27,
1040 platformSdkCodename: "OMR1",
1041 platformSdkFinal: false,
1042 expectedMinSdkVersion: "OMR1",
1043 },
1044 {
1045 name: "default final SDK",
1046 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001047 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001048 platformSdkInt: 27,
1049 platformSdkCodename: "REL",
1050 platformSdkFinal: true,
1051 expectedMinSdkVersion: "27",
1052 },
1053 {
1054 name: "default non-final SDK",
1055 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001056 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001057 platformSdkInt: 27,
1058 platformSdkCodename: "OMR1",
1059 platformSdkFinal: false,
1060 expectedMinSdkVersion: "OMR1",
1061 },
1062 {
1063 name: "14",
1064 sdkVersion: "14",
1065 expectedMinSdkVersion: "14",
1066 },
1067 }
1068
1069 for _, moduleType := range []string{"android_app", "android_library"} {
1070 for _, test := range testCases {
1071 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001072 platformApiProp := ""
1073 if test.platformApis {
1074 platformApiProp = "platform_apis: true,"
1075 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001076 bp := fmt.Sprintf(`%s {
1077 name: "foo",
1078 srcs: ["a.java"],
1079 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001080 %s
1081 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001082
Colin Cross98be1bb2019-12-13 20:41:13 -08001083 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001084 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1085 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1086 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1087
Colin Cross98be1bb2019-12-13 20:41:13 -08001088 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -07001089
1090 run(t, ctx, config)
1091
1092 foo := ctx.ModuleForTests("foo", "android_common")
1093 link := foo.Output("package-res.apk")
1094 linkFlags := strings.Split(link.Args["flags"], " ")
1095 min := android.IndexList("--min-sdk-version", linkFlags)
1096 target := android.IndexList("--target-sdk-version", linkFlags)
1097
1098 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1099 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1100 }
1101
1102 gotMinSdkVersion := linkFlags[min+1]
1103 gotTargetSdkVersion := linkFlags[target+1]
1104
1105 if gotMinSdkVersion != test.expectedMinSdkVersion {
1106 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1107 test.expectedMinSdkVersion, gotMinSdkVersion)
1108 }
1109
1110 if gotTargetSdkVersion != test.expectedMinSdkVersion {
1111 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1112 test.expectedMinSdkVersion, gotTargetSdkVersion)
1113 }
1114 })
1115 }
1116 }
1117}
Colin Crossa4f08812018-10-02 22:03:40 -07001118
Paul Duffin50c217c2019-06-12 13:25:22 +01001119func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001120 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001121 cc_library {
1122 name: "libjni",
1123 system_shared_libs: [],
Colin Cross01fd7cc2020-02-19 16:54:04 -08001124 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001125 stl: "none",
1126 }
1127
1128 android_test {
1129 name: "test",
1130 sdk_version: "core_platform",
1131 jni_libs: ["libjni"],
1132 }
1133
1134 android_test {
1135 name: "test_first",
1136 sdk_version: "core_platform",
1137 compile_multilib: "first",
1138 jni_libs: ["libjni"],
1139 }
1140
1141 android_test {
1142 name: "test_both",
1143 sdk_version: "core_platform",
1144 compile_multilib: "both",
1145 jni_libs: ["libjni"],
1146 }
1147
1148 android_test {
1149 name: "test_32",
1150 sdk_version: "core_platform",
1151 compile_multilib: "32",
1152 jni_libs: ["libjni"],
1153 }
1154
1155 android_test {
1156 name: "test_64",
1157 sdk_version: "core_platform",
1158 compile_multilib: "64",
1159 jni_libs: ["libjni"],
1160 }
1161 `)
1162
1163 testCases := []struct {
1164 name string
1165 abis []string
1166 }{
1167 {"test", []string{"arm64-v8a"}},
1168 {"test_first", []string{"arm64-v8a"}},
1169 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1170 {"test_32", []string{"armeabi-v7a"}},
1171 {"test_64", []string{"arm64-v8a"}},
1172 }
1173
1174 for _, test := range testCases {
1175 t.Run(test.name, func(t *testing.T) {
1176 app := ctx.ModuleForTests(test.name, "android_common")
1177 jniLibZip := app.Output("jnilibs.zip")
1178 var abis []string
1179 args := strings.Fields(jniLibZip.Args["jarArgs"])
1180 for i := 0; i < len(args); i++ {
1181 if args[i] == "-P" {
1182 abis = append(abis, filepath.Base(args[i+1]))
1183 i++
1184 }
1185 }
1186 if !reflect.DeepEqual(abis, test.abis) {
1187 t.Errorf("want abis %v, got %v", test.abis, abis)
1188 }
1189 })
1190 }
1191}
1192
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001193func TestAppSdkVersionByPartition(t *testing.T) {
1194 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1195 android_app {
1196 name: "foo",
1197 srcs: ["a.java"],
1198 vendor: true,
1199 platform_apis: true,
1200 }
1201 `)
1202
1203 testJava(t, `
1204 android_app {
1205 name: "bar",
1206 srcs: ["b.java"],
1207 platform_apis: true,
1208 }
1209 `)
1210
1211 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001212 bp := `
1213 android_app {
1214 name: "foo",
1215 srcs: ["a.java"],
1216 product_specific: true,
1217 platform_apis: true,
1218 }
1219 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001220
1221 config := testAppConfig(nil, bp, nil)
1222 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001223 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001224 testJavaErrorWithConfig(t, "sdk_version must have a value when the module is located at vendor or product", config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001225 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001226 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001227 }
1228 }
1229}
1230
Paul Duffin50c217c2019-06-12 13:25:22 +01001231func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001232 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001233 cc_library {
1234 name: "libjni",
1235 system_shared_libs: [],
1236 stl: "none",
Colin Cross1c93c292020-02-15 10:38:00 -08001237 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001238 }
1239
1240 android_app {
1241 name: "app",
1242 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001243 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001244 }
1245
1246 android_app {
1247 name: "app_noembed",
1248 jni_libs: ["libjni"],
1249 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001250 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001251 }
1252
1253 android_app {
1254 name: "app_embed",
1255 jni_libs: ["libjni"],
1256 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001257 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001258 }
1259
1260 android_test {
1261 name: "test",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001262 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001263 jni_libs: ["libjni"],
1264 }
1265
1266 android_test {
1267 name: "test_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001268 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001269 jni_libs: ["libjni"],
1270 use_embedded_native_libs: false,
1271 }
1272
1273 android_test_helper_app {
1274 name: "test_helper",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001275 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001276 jni_libs: ["libjni"],
1277 }
1278
1279 android_test_helper_app {
1280 name: "test_helper_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001281 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001282 jni_libs: ["libjni"],
1283 use_embedded_native_libs: false,
1284 }
1285 `)
1286
1287 testCases := []struct {
1288 name string
1289 packaged bool
1290 compressed bool
1291 }{
1292 {"app", false, false},
1293 {"app_noembed", false, false},
1294 {"app_embed", true, false},
1295 {"test", true, false},
1296 {"test_noembed", true, true},
1297 {"test_helper", true, false},
1298 {"test_helper_noembed", true, true},
1299 }
1300
1301 for _, test := range testCases {
1302 t.Run(test.name, func(t *testing.T) {
1303 app := ctx.ModuleForTests(test.name, "android_common")
1304 jniLibZip := app.MaybeOutput("jnilibs.zip")
1305 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1306 t.Errorf("expected jni packaged %v, got %v", w, g)
1307 }
1308
1309 if jniLibZip.Rule != nil {
1310 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1311 t.Errorf("expected jni compressed %v, got %v", w, g)
1312 }
Colin Cross01fd7cc2020-02-19 16:54:04 -08001313
1314 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1315 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1316 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001317 }
1318 })
1319 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001320}
1321
Colin Cross1dd9c442020-05-08 11:20:24 -07001322func TestJNISDK(t *testing.T) {
1323 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1324 cc_library {
1325 name: "libjni",
1326 system_shared_libs: [],
1327 stl: "none",
1328 sdk_version: "current",
1329 }
1330
1331 android_test {
1332 name: "app_platform",
1333 jni_libs: ["libjni"],
1334 platform_apis: true,
1335 }
1336
1337 android_test {
1338 name: "app_sdk",
1339 jni_libs: ["libjni"],
1340 sdk_version: "current",
1341 }
1342
1343 android_test {
1344 name: "app_force_platform",
1345 jni_libs: ["libjni"],
1346 sdk_version: "current",
1347 jni_uses_platform_apis: true,
1348 }
1349
1350 android_test {
1351 name: "app_force_sdk",
1352 jni_libs: ["libjni"],
1353 platform_apis: true,
1354 jni_uses_sdk_apis: true,
1355 }
Colin Crosseb032962020-05-13 11:05:02 -07001356
1357 cc_library {
1358 name: "libvendorjni",
1359 system_shared_libs: [],
1360 stl: "none",
1361 vendor: true,
1362 }
1363
1364 android_test {
1365 name: "app_vendor",
1366 jni_libs: ["libvendorjni"],
1367 sdk_version: "current",
1368 vendor: true,
1369 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001370 `)
1371
1372 testCases := []struct {
Colin Crosseb032962020-05-13 11:05:02 -07001373 name string
1374 sdkJNI bool
1375 vendorJNI bool
Colin Cross1dd9c442020-05-08 11:20:24 -07001376 }{
Colin Crosseb032962020-05-13 11:05:02 -07001377 {name: "app_platform"},
1378 {name: "app_sdk", sdkJNI: true},
1379 {name: "app_force_platform"},
1380 {name: "app_force_sdk", sdkJNI: true},
1381 {name: "app_vendor", vendorJNI: true},
Colin Cross1dd9c442020-05-08 11:20:24 -07001382 }
1383
Colin Crosseb032962020-05-13 11:05:02 -07001384 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1385 Output("libjni.so").Output.String()
1386 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1387 Output("libjni.so").Output.String()
1388 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1389 Output("libvendorjni.so").Output.String()
1390
Colin Cross1dd9c442020-05-08 11:20:24 -07001391 for _, test := range testCases {
1392 t.Run(test.name, func(t *testing.T) {
1393 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross1dd9c442020-05-08 11:20:24 -07001394
1395 jniLibZip := app.MaybeOutput("jnilibs.zip")
1396 if len(jniLibZip.Implicits) != 1 {
1397 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1398 }
1399 gotJNI := jniLibZip.Implicits[0].String()
1400
1401 if test.sdkJNI {
1402 if gotJNI != sdkJNI {
1403 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1404 }
Colin Crosseb032962020-05-13 11:05:02 -07001405 } else if test.vendorJNI {
1406 if gotJNI != vendorJNI {
1407 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1408 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001409 } else {
1410 if gotJNI != platformJNI {
1411 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1412 }
1413 }
1414 })
1415 }
1416
1417 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1418 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1419 android_test {
1420 name: "app_platform",
1421 platform_apis: true,
1422 jni_uses_platform_apis: true,
1423 }
1424 `)
1425 })
1426
1427 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1428 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1429 android_test {
1430 name: "app_sdk",
1431 sdk_version: "current",
1432 jni_uses_sdk_apis: true,
1433 }
1434 `)
1435 })
1436
1437}
1438
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001439func TestCertificates(t *testing.T) {
1440 testCases := []struct {
1441 name string
1442 bp string
1443 certificateOverride string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001444 expectedLineage string
1445 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001446 }{
1447 {
1448 name: "default",
1449 bp: `
1450 android_app {
1451 name: "foo",
1452 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001453 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001454 }
1455 `,
1456 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001457 expectedLineage: "",
1458 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001459 },
1460 {
1461 name: "module certificate property",
1462 bp: `
1463 android_app {
1464 name: "foo",
1465 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001466 certificate: ":new_certificate",
1467 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001468 }
1469
1470 android_app_certificate {
1471 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001472 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001473 }
1474 `,
1475 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001476 expectedLineage: "",
1477 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001478 },
1479 {
1480 name: "path certificate property",
1481 bp: `
1482 android_app {
1483 name: "foo",
1484 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001485 certificate: "expiredkey",
1486 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001487 }
1488 `,
1489 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001490 expectedLineage: "",
1491 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001492 },
1493 {
1494 name: "certificate overrides",
1495 bp: `
1496 android_app {
1497 name: "foo",
1498 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001499 certificate: "expiredkey",
1500 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001501 }
1502
1503 android_app_certificate {
1504 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001505 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001506 }
1507 `,
1508 certificateOverride: "foo:new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001509 expectedLineage: "",
1510 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1511 },
1512 {
1513 name: "certificate lineage",
1514 bp: `
1515 android_app {
1516 name: "foo",
1517 srcs: ["a.java"],
1518 certificate: ":new_certificate",
1519 lineage: "lineage.bin",
1520 sdk_version: "current",
1521 }
1522
1523 android_app_certificate {
1524 name: "new_certificate",
1525 certificate: "cert/new_cert",
1526 }
1527 `,
1528 certificateOverride: "",
1529 expectedLineage: "--lineage lineage.bin",
1530 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001531 },
1532 }
1533
1534 for _, test := range testCases {
1535 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001536 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001537 if test.certificateOverride != "" {
1538 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1539 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001540 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001541
1542 run(t, ctx, config)
1543 foo := ctx.ModuleForTests("foo", "android_common")
1544
1545 signapk := foo.Output("foo.apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001546 signCertificateFlags := signapk.Args["certificates"]
1547 if test.expectedCertificate != signCertificateFlags {
1548 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1549 }
1550
1551 signFlags := signapk.Args["flags"]
1552 if test.expectedLineage != signFlags {
1553 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001554 }
1555 })
1556 }
1557}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001558
Songchun Fan688de9a2020-03-24 20:32:24 -07001559func TestRequestV4SigningFlag(t *testing.T) {
1560 testCases := []struct {
1561 name string
1562 bp string
1563 expected string
1564 }{
1565 {
1566 name: "default",
1567 bp: `
1568 android_app {
1569 name: "foo",
1570 srcs: ["a.java"],
1571 sdk_version: "current",
1572 }
1573 `,
1574 expected: "",
1575 },
1576 {
1577 name: "default",
1578 bp: `
1579 android_app {
1580 name: "foo",
1581 srcs: ["a.java"],
1582 sdk_version: "current",
1583 v4_signature: false,
1584 }
1585 `,
1586 expected: "",
1587 },
1588 {
1589 name: "module certificate property",
1590 bp: `
1591 android_app {
1592 name: "foo",
1593 srcs: ["a.java"],
1594 sdk_version: "current",
1595 v4_signature: true,
1596 }
1597 `,
1598 expected: "--enable-v4",
1599 },
1600 }
1601
1602 for _, test := range testCases {
1603 t.Run(test.name, func(t *testing.T) {
1604 config := testAppConfig(nil, test.bp, nil)
1605 ctx := testContext()
1606
1607 run(t, ctx, config)
1608 foo := ctx.ModuleForTests("foo", "android_common")
1609
1610 signapk := foo.Output("foo.apk")
1611 signFlags := signapk.Args["flags"]
1612 if test.expected != signFlags {
1613 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1614 }
1615 })
1616 }
1617}
1618
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001619func TestPackageNameOverride(t *testing.T) {
1620 testCases := []struct {
1621 name string
1622 bp string
1623 packageNameOverride string
1624 expected []string
1625 }{
1626 {
1627 name: "default",
1628 bp: `
1629 android_app {
1630 name: "foo",
1631 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001632 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001633 }
1634 `,
1635 packageNameOverride: "",
1636 expected: []string{
1637 buildDir + "/.intermediates/foo/android_common/foo.apk",
1638 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1639 },
1640 },
1641 {
1642 name: "overridden",
1643 bp: `
1644 android_app {
1645 name: "foo",
1646 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001647 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001648 }
1649 `,
1650 packageNameOverride: "foo:bar",
1651 expected: []string{
1652 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001653 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001654 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1655 },
1656 },
1657 }
1658
1659 for _, test := range testCases {
1660 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001661 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001662 if test.packageNameOverride != "" {
1663 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1664 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001665 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001666
1667 run(t, ctx, config)
1668 foo := ctx.ModuleForTests("foo", "android_common")
1669
1670 outputs := foo.AllOutputs()
1671 outputMap := make(map[string]bool)
1672 for _, o := range outputs {
1673 outputMap[o] = true
1674 }
1675 for _, e := range test.expected {
1676 if _, exist := outputMap[e]; !exist {
1677 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1678 }
1679 }
1680 })
1681 }
1682}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001683
1684func TestInstrumentationTargetOverridden(t *testing.T) {
1685 bp := `
1686 android_app {
1687 name: "foo",
1688 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001689 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001690 }
1691
1692 android_test {
1693 name: "bar",
1694 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001695 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001696 }
1697 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001698 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001699 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001700 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001701
1702 run(t, ctx, config)
1703
1704 bar := ctx.ModuleForTests("bar", "android_common")
1705 res := bar.Output("package-res.apk")
1706 aapt2Flags := res.Args["flags"]
1707 e := "--rename-instrumentation-target-package org.dandroid.bp"
1708 if !strings.Contains(aapt2Flags, e) {
1709 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1710 }
1711}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001712
1713func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001714 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001715 android_app {
1716 name: "foo",
1717 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001718 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001719 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001720 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001721 }
1722
1723 override_android_app {
1724 name: "bar",
1725 base: "foo",
1726 certificate: ":new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001727 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001728 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001729 }
1730
1731 android_app_certificate {
1732 name: "new_certificate",
1733 certificate: "cert/new_cert",
1734 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001735
1736 override_android_app {
1737 name: "baz",
1738 base: "foo",
1739 package_name: "org.dandroid.bp",
1740 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001741 `)
1742
1743 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001744 moduleName string
1745 variantName string
1746 apkName string
1747 apkPath string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001748 certFlag string
1749 lineageFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001750 overrides []string
1751 aaptFlag string
1752 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001753 }{
1754 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001755 moduleName: "foo",
1756 variantName: "android_common",
1757 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001758 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1759 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001760 overrides: []string{"qux"},
1761 aaptFlag: "",
1762 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001763 },
1764 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001765 moduleName: "bar",
1766 variantName: "android_common_bar",
1767 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001768 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1769 lineageFlag: "--lineage lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001770 overrides: []string{"qux", "foo"},
1771 aaptFlag: "",
1772 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001773 },
1774 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001775 moduleName: "baz",
1776 variantName: "android_common_baz",
1777 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001778 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1779 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001780 overrides: []string{"qux", "foo"},
1781 aaptFlag: "--rename-manifest-package org.dandroid.bp",
1782 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001783 },
1784 }
1785 for _, expected := range expectedVariants {
1786 variant := ctx.ModuleForTests("foo", expected.variantName)
1787
1788 // Check the final apk name
1789 outputs := variant.AllOutputs()
1790 expectedApkPath := buildDir + expected.apkPath
1791 found := false
1792 for _, o := range outputs {
1793 if o == expectedApkPath {
1794 found = true
1795 break
1796 }
1797 }
1798 if !found {
1799 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1800 }
1801
1802 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001803 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001804 certFlag := signapk.Args["certificates"]
1805 if expected.certFlag != certFlag {
1806 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1807 }
1808
1809 // Check the lineage flags
1810 lineageFlag := signapk.Args["flags"]
1811 if expected.lineageFlag != lineageFlag {
1812 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001813 }
1814
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001815 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001816 mod := variant.Module().(*AndroidApp)
1817 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1818 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1819 expected.overrides, mod.appProperties.Overrides)
1820 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001821
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001822 // Test Overridable property: Logging_parent
1823 logging_parent := mod.aapt.LoggingParent
1824 if expected.logging_parent != logging_parent {
1825 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1826 expected.logging_parent, logging_parent)
1827 }
1828
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001829 // Check the package renaming flag, if exists.
1830 res := variant.Output("package-res.apk")
1831 aapt2Flags := res.Args["flags"]
1832 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1833 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1834 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001835 }
1836}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001837
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001838func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001839 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001840 android_app {
1841 name: "foo",
1842 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001843 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001844 }
1845
1846 override_android_app {
1847 name: "bar",
1848 base: "foo",
1849 package_name: "org.dandroid.bp",
1850 }
1851
1852 android_test {
1853 name: "baz",
1854 srcs: ["b.java"],
1855 instrumentation_for: "foo",
1856 }
1857
1858 android_test {
1859 name: "qux",
1860 srcs: ["b.java"],
1861 instrumentation_for: "bar",
1862 }
1863 `)
1864
1865 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1866 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1867 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1868 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1869 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1870 }
1871
1872 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1873 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1874 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1875 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1876 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1877 }
1878}
1879
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001880func TestOverrideAndroidTest(t *testing.T) {
1881 ctx, _ := testJava(t, `
1882 android_app {
1883 name: "foo",
1884 srcs: ["a.java"],
1885 package_name: "com.android.foo",
1886 sdk_version: "current",
1887 }
1888
1889 override_android_app {
1890 name: "bar",
1891 base: "foo",
1892 package_name: "com.android.bar",
1893 }
1894
1895 android_test {
1896 name: "foo_test",
1897 srcs: ["b.java"],
1898 instrumentation_for: "foo",
1899 }
1900
1901 override_android_test {
1902 name: "bar_test",
1903 base: "foo_test",
1904 package_name: "com.android.bar.test",
1905 instrumentation_for: "bar",
1906 instrumentation_target_package: "com.android.bar",
1907 }
1908 `)
1909
1910 expectedVariants := []struct {
1911 moduleName string
1912 variantName string
1913 apkPath string
1914 overrides []string
1915 targetVariant string
1916 packageFlag string
1917 targetPackageFlag string
1918 }{
1919 {
1920 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001921 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001922 overrides: nil,
1923 targetVariant: "android_common",
1924 packageFlag: "",
1925 targetPackageFlag: "",
1926 },
1927 {
1928 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001929 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001930 overrides: []string{"foo_test"},
1931 targetVariant: "android_common_bar",
1932 packageFlag: "com.android.bar.test",
1933 targetPackageFlag: "com.android.bar",
1934 },
1935 }
1936 for _, expected := range expectedVariants {
1937 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1938
1939 // Check the final apk name
1940 outputs := variant.AllOutputs()
1941 expectedApkPath := buildDir + expected.apkPath
1942 found := false
1943 for _, o := range outputs {
1944 if o == expectedApkPath {
1945 found = true
1946 break
1947 }
1948 }
1949 if !found {
1950 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1951 }
1952
1953 // Check if the overrides field values are correctly aggregated.
1954 mod := variant.Module().(*AndroidTest)
1955 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1956 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1957 expected.overrides, mod.appProperties.Overrides)
1958 }
1959
1960 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1961 javac := variant.Rule("javac")
1962 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1963 if !strings.Contains(javac.Args["classpath"], turbine) {
1964 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1965 }
1966
1967 // Check aapt2 flags.
1968 res := variant.Output("package-res.apk")
1969 aapt2Flags := res.Args["flags"]
1970 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1971 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1972 }
1973}
1974
Jaewoong Jung39982342020-01-14 10:27:18 -08001975func TestAndroidTest_FixTestConfig(t *testing.T) {
1976 ctx, _ := testJava(t, `
1977 android_app {
1978 name: "foo",
1979 srcs: ["a.java"],
1980 package_name: "com.android.foo",
1981 sdk_version: "current",
1982 }
1983
1984 android_test {
1985 name: "foo_test",
1986 srcs: ["b.java"],
1987 instrumentation_for: "foo",
1988 }
1989
1990 android_test {
1991 name: "bar_test",
1992 srcs: ["b.java"],
1993 package_name: "com.android.bar.test",
1994 instrumentation_for: "foo",
1995 }
1996
1997 override_android_test {
1998 name: "baz_test",
1999 base: "foo_test",
2000 package_name: "com.android.baz.test",
2001 }
2002 `)
2003
2004 testCases := []struct {
2005 moduleName string
2006 variantName string
2007 expectedFlags []string
2008 }{
2009 {
2010 moduleName: "foo_test",
2011 variantName: "android_common",
2012 },
2013 {
2014 moduleName: "bar_test",
2015 variantName: "android_common",
2016 expectedFlags: []string{
2017 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
2018 "--package-name com.android.bar.test",
2019 },
2020 },
2021 {
2022 moduleName: "foo_test",
2023 variantName: "android_common_baz_test",
2024 expectedFlags: []string{
2025 "--manifest " + buildDir +
2026 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
2027 "--package-name com.android.baz.test",
2028 "--test-file-name baz_test.apk",
2029 },
2030 },
2031 }
2032
2033 for _, test := range testCases {
2034 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
2035 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
2036
2037 if len(test.expectedFlags) > 0 {
2038 if params.Rule == nil {
2039 t.Errorf("test_config_fixer was expected to run, but didn't")
2040 } else {
2041 for _, flag := range test.expectedFlags {
2042 if !strings.Contains(params.RuleParams.Command, flag) {
2043 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2044 }
2045 }
2046 }
2047 } else {
2048 if params.Rule != nil {
2049 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2050 }
2051 }
2052
2053 }
2054}
2055
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002056func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002057 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002058 android_app_import {
2059 name: "foo",
2060 apk: "prebuilts/apk/app.apk",
2061 certificate: "platform",
2062 dex_preopt: {
2063 enabled: true,
2064 },
2065 }
2066 `)
2067
2068 variant := ctx.ModuleForTests("foo", "android_common")
2069
2070 // Check dexpreopt outputs.
2071 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2072 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2073 t.Errorf("can't find dexpreopt outputs")
2074 }
2075
2076 // Check cert signing flag.
2077 signedApk := variant.Output("signed/foo.apk")
2078 signingFlag := signedApk.Args["certificates"]
2079 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2080 if expected != signingFlag {
2081 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2082 }
2083}
2084
2085func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002086 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002087 android_app_import {
2088 name: "foo",
2089 apk: "prebuilts/apk/app.apk",
2090 certificate: "platform",
2091 dex_preopt: {
2092 enabled: false,
2093 },
2094 }
2095 `)
2096
2097 variant := ctx.ModuleForTests("foo", "android_common")
2098
2099 // Check dexpreopt outputs. They shouldn't exist.
2100 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2101 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2102 t.Errorf("dexpreopt shouldn't have run.")
2103 }
2104}
2105
2106func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002107 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002108 android_app_import {
2109 name: "foo",
2110 apk: "prebuilts/apk/app.apk",
2111 presigned: true,
2112 dex_preopt: {
2113 enabled: true,
2114 },
2115 }
2116 `)
2117
2118 variant := ctx.ModuleForTests("foo", "android_common")
2119
2120 // Check dexpreopt outputs.
2121 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2122 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2123 t.Errorf("can't find dexpreopt outputs")
2124 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002125 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002126 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2127 t.Errorf("signing rule shouldn't be included.")
2128 }
2129 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2130 t.Errorf("can't find aligning rule")
2131 }
2132}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002133
Liz Kammer2bc57f62020-05-13 15:49:21 -07002134func TestAndroidAppImport_SigningLineage(t *testing.T) {
2135 ctx, _ := testJava(t, `
2136 android_app_import {
2137 name: "foo",
2138 apk: "prebuilts/apk/app.apk",
2139 certificate: "platform",
2140 lineage: "lineage.bin",
2141 }
2142 `)
2143
2144 variant := ctx.ModuleForTests("foo", "android_common")
2145
2146 // Check cert signing lineage flag.
2147 signedApk := variant.Output("signed/foo.apk")
2148 signingFlag := signedApk.Args["flags"]
2149 expected := "--lineage lineage.bin"
2150 if expected != signingFlag {
2151 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2152 }
2153}
2154
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002155func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2156 ctx, _ := testJava(t, `
2157 android_app_import {
2158 name: "foo",
2159 apk: "prebuilts/apk/app.apk",
2160 default_dev_cert: true,
2161 dex_preopt: {
2162 enabled: true,
2163 },
2164 }
2165 `)
2166
2167 variant := ctx.ModuleForTests("foo", "android_common")
2168
2169 // Check dexpreopt outputs.
2170 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2171 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2172 t.Errorf("can't find dexpreopt outputs")
2173 }
2174
2175 // Check cert signing flag.
2176 signedApk := variant.Output("signed/foo.apk")
2177 signingFlag := signedApk.Args["certificates"]
2178 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2179 if expected != signingFlag {
2180 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2181 }
2182}
2183
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002184func TestAndroidAppImport_DpiVariants(t *testing.T) {
2185 bp := `
2186 android_app_import {
2187 name: "foo",
2188 apk: "prebuilts/apk/app.apk",
2189 dpi_variants: {
2190 xhdpi: {
2191 apk: "prebuilts/apk/app_xhdpi.apk",
2192 },
2193 xxhdpi: {
2194 apk: "prebuilts/apk/app_xxhdpi.apk",
2195 },
2196 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002197 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002198 dex_preopt: {
2199 enabled: true,
2200 },
2201 }
2202 `
2203 testCases := []struct {
2204 name string
2205 aaptPreferredConfig *string
2206 aaptPrebuiltDPI []string
2207 expected string
2208 }{
2209 {
2210 name: "no preferred",
2211 aaptPreferredConfig: nil,
2212 aaptPrebuiltDPI: []string{},
2213 expected: "prebuilts/apk/app.apk",
2214 },
2215 {
2216 name: "AAPTPreferredConfig matches",
2217 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002218 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002219 expected: "prebuilts/apk/app_xhdpi.apk",
2220 },
2221 {
2222 name: "AAPTPrebuiltDPI matches",
2223 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2224 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2225 expected: "prebuilts/apk/app_xxhdpi.apk",
2226 },
2227 {
2228 name: "non-first AAPTPrebuiltDPI matches",
2229 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2230 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2231 expected: "prebuilts/apk/app_xhdpi.apk",
2232 },
2233 {
2234 name: "no matches",
2235 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2236 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2237 expected: "prebuilts/apk/app.apk",
2238 },
2239 }
2240
2241 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2242 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002243 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002244 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2245 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002246 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002247
2248 run(t, ctx, config)
2249
2250 variant := ctx.ModuleForTests("foo", "android_common")
2251 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2252 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2253 if len(matches) != 2 {
2254 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2255 }
2256 if test.expected != matches[1] {
2257 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2258 }
2259 }
2260}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002261
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002262func TestAndroidAppImport_Filename(t *testing.T) {
2263 ctx, config := testJava(t, `
2264 android_app_import {
2265 name: "foo",
2266 apk: "prebuilts/apk/app.apk",
2267 presigned: true,
2268 }
2269
2270 android_app_import {
2271 name: "bar",
2272 apk: "prebuilts/apk/app.apk",
2273 presigned: true,
2274 filename: "bar_sample.apk"
2275 }
2276 `)
2277
2278 testCases := []struct {
2279 name string
2280 expected string
2281 }{
2282 {
2283 name: "foo",
2284 expected: "foo.apk",
2285 },
2286 {
2287 name: "bar",
2288 expected: "bar_sample.apk",
2289 },
2290 }
2291
2292 for _, test := range testCases {
2293 variant := ctx.ModuleForTests(test.name, "android_common")
2294 if variant.MaybeOutput(test.expected).Rule == nil {
2295 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2296 }
2297
2298 a := variant.Module().(*AndroidAppImport)
2299 expectedValues := []string{test.expected}
2300 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002301 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002302 if !reflect.DeepEqual(actualValues, expectedValues) {
2303 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2304 actualValues, expectedValues)
2305 }
2306 }
2307}
2308
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002309func TestAndroidAppImport_ArchVariants(t *testing.T) {
2310 // The test config's target arch is ARM64.
2311 testCases := []struct {
2312 name string
2313 bp string
2314 expected string
2315 }{
2316 {
2317 name: "matching arch",
2318 bp: `
2319 android_app_import {
2320 name: "foo",
2321 apk: "prebuilts/apk/app.apk",
2322 arch: {
2323 arm64: {
2324 apk: "prebuilts/apk/app_arm64.apk",
2325 },
2326 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002327 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002328 dex_preopt: {
2329 enabled: true,
2330 },
2331 }
2332 `,
2333 expected: "prebuilts/apk/app_arm64.apk",
2334 },
2335 {
2336 name: "no matching arch",
2337 bp: `
2338 android_app_import {
2339 name: "foo",
2340 apk: "prebuilts/apk/app.apk",
2341 arch: {
2342 arm: {
2343 apk: "prebuilts/apk/app_arm.apk",
2344 },
2345 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002346 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002347 dex_preopt: {
2348 enabled: true,
2349 },
2350 }
2351 `,
2352 expected: "prebuilts/apk/app.apk",
2353 },
2354 }
2355
2356 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2357 for _, test := range testCases {
2358 ctx, _ := testJava(t, test.bp)
2359
2360 variant := ctx.ModuleForTests("foo", "android_common")
2361 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2362 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2363 if len(matches) != 2 {
2364 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2365 }
2366 if test.expected != matches[1] {
2367 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2368 }
2369 }
2370}
2371
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002372func TestAndroidTestImport(t *testing.T) {
2373 ctx, config := testJava(t, `
2374 android_test_import {
2375 name: "foo",
2376 apk: "prebuilts/apk/app.apk",
2377 presigned: true,
2378 data: [
2379 "testdata/data",
2380 ],
2381 }
2382 `)
2383
2384 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2385
2386 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002387 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002388 expected := []string{"tests"}
2389 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2390 if !reflect.DeepEqual(expected, actual) {
2391 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2392 }
2393 expected = []string{"testdata/data:testdata/data"}
2394 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2395 if !reflect.DeepEqual(expected, actual) {
2396 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2397 }
2398}
2399
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002400func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2401 ctx, _ := testJava(t, `
2402 android_test_import {
2403 name: "foo",
2404 apk: "prebuilts/apk/app.apk",
2405 certificate: "cert/new_cert",
2406 data: [
2407 "testdata/data",
2408 ],
2409 }
2410
2411 android_test_import {
2412 name: "foo_presigned",
2413 apk: "prebuilts/apk/app.apk",
2414 presigned: true,
2415 data: [
2416 "testdata/data",
2417 ],
2418 }
2419 `)
2420
2421 variant := ctx.ModuleForTests("foo", "android_common")
2422 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2423 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2424 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2425 }
2426
2427 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2428 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2429 if jniRule != android.Cp.String() {
2430 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2431 }
Liz Kammer7e20dda2020-05-20 14:36:30 -07002432 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2433 t.Errorf("Presigned test apk should be aligned")
2434 }
2435}
2436
2437func TestAndroidTestImport_Preprocessed(t *testing.T) {
2438 ctx, _ := testJava(t, `
2439 android_test_import {
2440 name: "foo",
2441 apk: "prebuilts/apk/app.apk",
2442 presigned: true,
2443 preprocessed: true,
2444 }
2445
2446 android_test_import {
2447 name: "foo_cert",
2448 apk: "prebuilts/apk/app.apk",
2449 certificate: "cert/new_cert",
2450 preprocessed: true,
2451 }
2452 `)
2453
2454 testModules := []string{"foo", "foo_cert"}
2455 for _, m := range testModules {
2456 apkName := m + ".apk"
2457 variant := ctx.ModuleForTests(m, "android_common")
2458 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2459 if jniRule != android.Cp.String() {
2460 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2461 }
2462
2463 // Make sure signing and aligning were skipped.
2464 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2465 t.Errorf("signing rule shouldn't be included for preprocessed.")
2466 }
2467 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2468 t.Errorf("aligning rule shouldn't be for preprocessed")
2469 }
2470 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002471}
2472
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002473func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002474 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002475 cc_library {
2476 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002477 sdk_version: "current",
2478 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002479 }
2480
2481 android_test {
2482 name: "stl",
2483 jni_libs: ["libjni"],
2484 compile_multilib: "both",
2485 sdk_version: "current",
2486 stl: "c++_shared",
2487 }
2488
2489 android_test {
2490 name: "system",
2491 jni_libs: ["libjni"],
2492 compile_multilib: "both",
2493 sdk_version: "current",
2494 }
2495 `)
2496
2497 testCases := []struct {
2498 name string
2499 jnis []string
2500 }{
2501 {"stl",
2502 []string{
2503 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002504 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002505 },
2506 },
2507 {"system",
2508 []string{
2509 "libjni.so",
2510 },
2511 },
2512 }
2513
2514 for _, test := range testCases {
2515 t.Run(test.name, func(t *testing.T) {
2516 app := ctx.ModuleForTests(test.name, "android_common")
2517 jniLibZip := app.Output("jnilibs.zip")
2518 var jnis []string
2519 args := strings.Fields(jniLibZip.Args["jarArgs"])
2520 for i := 0; i < len(args); i++ {
2521 if args[i] == "-f" {
2522 jnis = append(jnis, args[i+1])
2523 i += 1
2524 }
2525 }
2526 jnisJoined := strings.Join(jnis, " ")
2527 for _, jni := range test.jnis {
2528 if !strings.Contains(jnisJoined, jni) {
2529 t.Errorf("missing jni %q in %q", jni, jnis)
2530 }
2531 }
2532 })
2533 }
2534}
Colin Cross50ddcc42019-05-16 12:28:22 -07002535
2536func TestUsesLibraries(t *testing.T) {
2537 bp := `
2538 java_sdk_library {
2539 name: "foo",
2540 srcs: ["a.java"],
2541 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002542 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002543 }
2544
2545 java_sdk_library {
Paul Duffin64e61992020-05-15 10:20:31 +01002546 name: "qux",
2547 srcs: ["a.java"],
2548 api_packages: ["qux"],
2549 sdk_version: "current",
2550 }
2551
2552 java_sdk_library {
2553 name: "quuz",
2554 srcs: ["a.java"],
2555 api_packages: ["quuz"],
2556 sdk_version: "current",
2557 }
2558
2559 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002560 name: "bar",
2561 srcs: ["a.java"],
2562 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002563 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002564 }
2565
2566 android_app {
2567 name: "app",
2568 srcs: ["a.java"],
Paul Duffin64e61992020-05-15 10:20:31 +01002569 libs: ["qux", "quuz.stubs"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002570 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002571 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002572 optional_uses_libs: [
2573 "bar",
2574 "baz",
2575 ],
2576 }
2577
2578 android_app_import {
2579 name: "prebuilt",
2580 apk: "prebuilts/apk/app.apk",
2581 certificate: "platform",
2582 uses_libs: ["foo"],
2583 optional_uses_libs: [
2584 "bar",
2585 "baz",
2586 ],
2587 }
2588 `
2589
Colin Cross98be1bb2019-12-13 20:41:13 -08002590 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002591 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2592
Colin Cross98be1bb2019-12-13 20:41:13 -08002593 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002594
2595 run(t, ctx, config)
2596
2597 app := ctx.ModuleForTests("app", "android_common")
2598 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2599
Paul Duffin64e61992020-05-15 10:20:31 +01002600 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2601 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2602 if w := "--uses-library qux"; !strings.Contains(manifestFixerArgs, w) {
2603 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2604 }
2605 if w := "--uses-library quuz"; !strings.Contains(manifestFixerArgs, w) {
2606 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2607 }
2608
Colin Cross50ddcc42019-05-16 12:28:22 -07002609 // Test that all libraries are verified
2610 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2611 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2612 t.Errorf("wanted %q in %q", w, cmd)
2613 }
2614
2615 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2616 t.Errorf("wanted %q in %q", w, cmd)
2617 }
2618
2619 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2620
2621 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2622 t.Errorf("wanted %q in %q", w, cmd)
2623 }
2624
2625 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2626 t.Errorf("wanted %q in %q", w, cmd)
2627 }
2628
2629 // Test that only present libraries are preopted
2630 cmd = app.Rule("dexpreopt").RuleParams.Command
2631
2632 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2633 t.Errorf("wanted %q in %q", w, cmd)
2634 }
2635
2636 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2637
2638 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2639 t.Errorf("wanted %q in %q", w, cmd)
2640 }
2641}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002642
2643func TestCodelessApp(t *testing.T) {
2644 testCases := []struct {
2645 name string
2646 bp string
2647 noCode bool
2648 }{
2649 {
2650 name: "normal",
2651 bp: `
2652 android_app {
2653 name: "foo",
2654 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002655 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002656 }
2657 `,
2658 noCode: false,
2659 },
2660 {
2661 name: "app without sources",
2662 bp: `
2663 android_app {
2664 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002665 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002666 }
2667 `,
2668 noCode: true,
2669 },
2670 {
2671 name: "app with libraries",
2672 bp: `
2673 android_app {
2674 name: "foo",
2675 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002676 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002677 }
2678
2679 java_library {
2680 name: "lib",
2681 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002682 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002683 }
2684 `,
2685 noCode: false,
2686 },
2687 {
2688 name: "app with sourceless libraries",
2689 bp: `
2690 android_app {
2691 name: "foo",
2692 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002693 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002694 }
2695
2696 java_library {
2697 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002698 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002699 }
2700 `,
2701 // TODO(jungjw): this should probably be true
2702 noCode: false,
2703 },
2704 }
2705
2706 for _, test := range testCases {
2707 t.Run(test.name, func(t *testing.T) {
2708 ctx := testApp(t, test.bp)
2709
2710 foo := ctx.ModuleForTests("foo", "android_common")
2711 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2712 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2713 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2714 }
2715 })
2716 }
2717}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002718
2719func TestEmbedNotice(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002720 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002721 android_app {
2722 name: "foo",
2723 srcs: ["a.java"],
2724 static_libs: ["javalib"],
2725 jni_libs: ["libjni"],
2726 notice: "APP_NOTICE",
2727 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002728 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002729 }
2730
2731 // No embed_notice flag
2732 android_app {
2733 name: "bar",
2734 srcs: ["a.java"],
2735 jni_libs: ["libjni"],
2736 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002737 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002738 }
2739
2740 // No NOTICE files
2741 android_app {
2742 name: "baz",
2743 srcs: ["a.java"],
2744 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002745 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002746 }
2747
2748 cc_library {
2749 name: "libjni",
2750 system_shared_libs: [],
2751 stl: "none",
2752 notice: "LIB_NOTICE",
Colin Cross1c93c292020-02-15 10:38:00 -08002753 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002754 }
2755
2756 java_library {
2757 name: "javalib",
2758 srcs: [
2759 ":gen",
2760 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002761 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002762 }
2763
2764 genrule {
2765 name: "gen",
2766 tools: ["gentool"],
2767 out: ["gen.java"],
2768 notice: "GENRULE_NOTICE",
2769 }
2770
2771 java_binary_host {
2772 name: "gentool",
2773 srcs: ["b.java"],
2774 notice: "TOOL_NOTICE",
2775 }
2776 `)
2777
2778 // foo has NOTICE files to process, and embed_notices is true.
2779 foo := ctx.ModuleForTests("foo", "android_common")
2780 // verify merge notices rule.
2781 mergeNotices := foo.Rule("mergeNoticesRule")
2782 noticeInputs := mergeNotices.Inputs.Strings()
2783 // TOOL_NOTICE should be excluded as it's a host module.
2784 if len(mergeNotices.Inputs) != 3 {
2785 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2786 }
2787 if !inList("APP_NOTICE", noticeInputs) {
2788 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2789 }
2790 if !inList("LIB_NOTICE", noticeInputs) {
2791 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2792 }
2793 if !inList("GENRULE_NOTICE", noticeInputs) {
2794 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2795 }
2796 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2797 res := foo.Output("package-res.apk")
2798 aapt2Flags := res.Args["flags"]
2799 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2800 if !strings.Contains(aapt2Flags, e) {
2801 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2802 }
2803
2804 // bar has NOTICE files to process, but embed_notices is not set.
2805 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002806 res = bar.Output("package-res.apk")
2807 aapt2Flags = res.Args["flags"]
2808 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2809 if strings.Contains(aapt2Flags, e) {
2810 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002811 }
2812
2813 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2814 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002815 res = baz.Output("package-res.apk")
2816 aapt2Flags = res.Args["flags"]
2817 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2818 if strings.Contains(aapt2Flags, e) {
2819 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002820 }
2821}
Colin Cross53a87f52019-06-25 13:35:30 -07002822
2823func TestUncompressDex(t *testing.T) {
2824 testCases := []struct {
2825 name string
2826 bp string
2827
2828 uncompressedPlatform bool
2829 uncompressedUnbundled bool
2830 }{
2831 {
2832 name: "normal",
2833 bp: `
2834 android_app {
2835 name: "foo",
2836 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002837 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002838 }
2839 `,
2840 uncompressedPlatform: true,
2841 uncompressedUnbundled: false,
2842 },
2843 {
2844 name: "use_embedded_dex",
2845 bp: `
2846 android_app {
2847 name: "foo",
2848 use_embedded_dex: true,
2849 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002850 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002851 }
2852 `,
2853 uncompressedPlatform: true,
2854 uncompressedUnbundled: true,
2855 },
2856 {
2857 name: "privileged",
2858 bp: `
2859 android_app {
2860 name: "foo",
2861 privileged: true,
2862 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002863 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002864 }
2865 `,
2866 uncompressedPlatform: true,
2867 uncompressedUnbundled: true,
2868 },
David Srbecky98c71222020-05-20 22:20:28 +01002869 {
2870 name: "normal_uncompress_dex_true",
2871 bp: `
2872 android_app {
2873 name: "foo",
2874 srcs: ["a.java"],
2875 sdk_version: "current",
2876 uncompress_dex: true,
2877 }
2878 `,
2879 uncompressedPlatform: true,
2880 uncompressedUnbundled: true,
2881 },
2882 {
2883 name: "normal_uncompress_dex_false",
2884 bp: `
2885 android_app {
2886 name: "foo",
2887 srcs: ["a.java"],
2888 sdk_version: "current",
2889 uncompress_dex: false,
2890 }
2891 `,
2892 uncompressedPlatform: false,
2893 uncompressedUnbundled: false,
2894 },
Colin Cross53a87f52019-06-25 13:35:30 -07002895 }
2896
2897 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2898 t.Helper()
2899
Colin Cross98be1bb2019-12-13 20:41:13 -08002900 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002901 if unbundled {
2902 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2903 }
2904
Colin Cross98be1bb2019-12-13 20:41:13 -08002905 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002906
2907 run(t, ctx, config)
2908
2909 foo := ctx.ModuleForTests("foo", "android_common")
2910 dex := foo.Rule("r8")
2911 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2912 aligned := foo.MaybeRule("zipalign").Rule != nil
2913
2914 if uncompressedInDexJar != want {
2915 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2916 }
2917
2918 if aligned != want {
2919 t.Errorf("want aligned %v, got %v", want, aligned)
2920 }
2921 }
2922
2923 for _, tt := range testCases {
2924 t.Run(tt.name, func(t *testing.T) {
2925 t.Run("platform", func(t *testing.T) {
2926 test(t, tt.bp, tt.uncompressedPlatform, false)
2927 })
2928 t.Run("unbundled", func(t *testing.T) {
2929 test(t, tt.bp, tt.uncompressedUnbundled, true)
2930 })
2931 })
2932 }
2933}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002934
2935func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2936 if expectedValue != "" {
2937 expectedFlag := "--" + flagName + " " + expectedValue
2938 if !strings.Contains(aapt2Flags, expectedFlag) {
2939 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2940 }
2941 } else {
2942 unexpectedFlag := "--" + flagName
2943 if strings.Contains(aapt2Flags, unexpectedFlag) {
2944 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2945 }
2946 }
2947}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002948
2949func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungca095d72020-04-09 16:15:30 -07002950 fs := map[string][]byte{
2951 "baz/res/res/values/strings.xml": nil,
2952 "bar/res/res/values/strings.xml": nil,
2953 }
2954 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002955 runtime_resource_overlay {
2956 name: "foo",
2957 certificate: "platform",
Liz Kammer7fe241f2020-05-19 16:15:25 -07002958 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002959 product_specific: true,
Jaewoong Jungca095d72020-04-09 16:15:30 -07002960 static_libs: ["bar"],
2961 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002962 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002963 }
2964
2965 runtime_resource_overlay {
2966 name: "foo_themed",
2967 certificate: "platform",
2968 product_specific: true,
2969 theme: "faza",
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002970 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002971 }
Jaewoong Jungca095d72020-04-09 16:15:30 -07002972
2973 android_library {
2974 name: "bar",
2975 resource_dirs: ["bar/res"],
2976 }
2977
2978 android_app {
2979 name: "baz",
2980 sdk_version: "current",
2981 resource_dirs: ["baz/res"],
2982 }
2983 `
2984 config := testAppConfig(nil, bp, fs)
2985 ctx := testContext()
2986 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002987
2988 m := ctx.ModuleForTests("foo", "android_common")
2989
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002990 // Check AAPT2 link flags.
2991 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2992 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2993 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2994 if len(absentFlags) > 0 {
2995 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2996 }
2997
Jaewoong Jungca095d72020-04-09 16:15:30 -07002998 // Check overlay.list output for static_libs dependency.
2999 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
3000 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
3001 if !inList(staticLibPackage, overlayList) {
3002 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
3003 }
3004
3005 // Check AAPT2 link flags for resource_libs dependency.
3006 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
3007 if !strings.Contains(aapt2Flags, resourceLibFlag) {
3008 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
3009 }
3010
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003011 // Check cert signing flag.
3012 signedApk := m.Output("signed/foo.apk")
Liz Kammer7fe241f2020-05-19 16:15:25 -07003013 lineageFlag := signedApk.Args["flags"]
3014 expectedLineageFlag := "--lineage lineage.bin"
3015 if expectedLineageFlag != lineageFlag {
3016 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
3017 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003018 signingFlag := signedApk.Args["certificates"]
3019 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
3020 if expected != signingFlag {
3021 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
3022 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003023 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3024 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003025 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
3026 if !reflect.DeepEqual(path, expectedPath) {
3027 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
3028 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003029
3030 // Check device location.
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003031 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003032 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003033 if !reflect.DeepEqual(path, expectedPath) {
3034 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3035 }
3036
3037 // A themed module has a different device location
3038 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003039 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3040 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003041 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3042 if !reflect.DeepEqual(path, expectedPath) {
3043 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3044 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003045
3046 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3047 expectedOverrides := []string{"foo"}
3048 if !reflect.DeepEqual(overrides, expectedOverrides) {
3049 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3050 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003051}
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07003052
3053func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3054 ctx, config := testJava(t, `
3055 java_defaults {
3056 name: "rro_defaults",
3057 theme: "default_theme",
3058 product_specific: true,
3059 aaptflags: ["--keep-raw-values"],
3060 }
3061
3062 runtime_resource_overlay {
3063 name: "foo_with_defaults",
3064 defaults: ["rro_defaults"],
3065 }
3066
3067 runtime_resource_overlay {
3068 name: "foo_barebones",
3069 }
3070 `)
3071
3072 //
3073 // RRO module with defaults
3074 //
3075 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3076
3077 // Check AAPT2 link flags.
3078 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3079 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3080 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3081 if len(absentFlags) > 0 {
3082 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3083 }
3084
3085 // Check device location.
3086 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3087 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3088 if !reflect.DeepEqual(path, expectedPath) {
3089 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3090 }
3091
3092 //
3093 // RRO module without defaults
3094 //
3095 m = ctx.ModuleForTests("foo_barebones", "android_common")
3096
3097 // Check AAPT2 link flags.
3098 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3099 unexpectedFlags := "--keep-raw-values"
3100 if inList(unexpectedFlags, aapt2Flags) {
3101 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3102 }
3103
3104 // Check device location.
3105 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3106 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3107 if !reflect.DeepEqual(path, expectedPath) {
3108 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3109 }
3110}
Roshan Piusb8307962020-04-27 09:42:27 -07003111
3112func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3113 ctx, _ := testJava(t, `
3114 runtime_resource_overlay {
3115 name: "foo_overlay",
3116 certificate: "platform",
3117 product_specific: true,
3118 sdk_version: "current",
3119 }
3120
3121 override_runtime_resource_overlay {
3122 name: "bar_overlay",
3123 base: "foo_overlay",
3124 package_name: "com.android.bar.overlay",
3125 target_package_name: "com.android.bar",
3126 }
3127 `)
3128
3129 expectedVariants := []struct {
3130 moduleName string
3131 variantName string
3132 apkPath string
3133 overrides []string
3134 targetVariant string
3135 packageFlag string
3136 targetPackageFlag string
3137 }{
3138 {
3139 variantName: "android_common",
3140 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3141 overrides: nil,
3142 targetVariant: "android_common",
3143 packageFlag: "",
3144 targetPackageFlag: "",
3145 },
3146 {
3147 variantName: "android_common_bar_overlay",
3148 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3149 overrides: []string{"foo_overlay"},
3150 targetVariant: "android_common_bar",
3151 packageFlag: "com.android.bar.overlay",
3152 targetPackageFlag: "com.android.bar",
3153 },
3154 }
3155 for _, expected := range expectedVariants {
3156 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3157
3158 // Check the final apk name
3159 outputs := variant.AllOutputs()
3160 expectedApkPath := buildDir + expected.apkPath
3161 found := false
3162 for _, o := range outputs {
3163 if o == expectedApkPath {
3164 found = true
3165 break
3166 }
3167 }
3168 if !found {
3169 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3170 }
3171
3172 // Check if the overrides field values are correctly aggregated.
3173 mod := variant.Module().(*RuntimeResourceOverlay)
3174 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3175 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3176 expected.overrides, mod.properties.Overrides)
3177 }
3178
3179 // Check aapt2 flags.
3180 res := variant.Output("package-res.apk")
3181 aapt2Flags := res.Args["flags"]
3182 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
3183 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3184 }
3185}