blob: 1123d84d90a7d3018adc2a768e5b15f3b3cfeafc [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 {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700176 name string
177 targets []android.Target
178 aaptPrebuiltDPI []string
179 sdkVersion int
180 expected map[string]string
Sasha Smundak4de27a52020-04-23 09:49:59 -0700181 }{
182 {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700183 name: "One",
184 targets: []android.Target{
185 {Os: android.Android, Arch: android.Arch{ArchType: android.X86}},
186 },
Sasha Smundak4de27a52020-04-23 09:49:59 -0700187 aaptPrebuiltDPI: []string{"ldpi", "xxhdpi"},
188 sdkVersion: 29,
189 expected: map[string]string{
190 "abis": "X86",
191 "allow-prereleased": "false",
192 "screen-densities": "LDPI,XXHDPI",
193 "sdk-version": "29",
194 "stem": "foo",
195 },
196 },
197 {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700198 name: "Two",
199 targets: []android.Target{
200 {Os: android.Android, Arch: android.Arch{ArchType: android.X86_64}},
201 {Os: android.Android, Arch: android.Arch{ArchType: android.X86}},
202 },
203 aaptPrebuiltDPI: nil,
204 sdkVersion: 30,
Sasha Smundak4de27a52020-04-23 09:49:59 -0700205 expected: map[string]string{
206 "abis": "X86_64,X86",
207 "allow-prereleased": "false",
208 "screen-densities": "all",
209 "sdk-version": "30",
210 "stem": "foo",
211 },
212 },
213 }
214
215 for _, test := range testCases {
216 config := testAppConfig(nil, bp, nil)
217 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
218 config.TestProductVariables.Platform_sdk_version = &test.sdkVersion
Jaewoong Jung829b7132020-06-10 12:23:32 -0700219 config.Targets[android.Android] = test.targets
Sasha Smundak4de27a52020-04-23 09:49:59 -0700220 ctx := testContext()
221 run(t, ctx, config)
222 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700223 const packedSplitApks = "foo.zip"
Sasha Smundak4de27a52020-04-23 09:49:59 -0700224 params := module.Output(packedSplitApks)
225 for k, v := range test.expected {
226 if actual := params.Args[k]; actual != v {
227 t.Errorf("%s: bad build arg value for '%s': '%s', expected '%s'",
228 test.name, k, actual, v)
229 }
230 }
231 }
232}
233
Jeongik Cha538c0d02019-07-11 15:54:27 +0900234func TestPlatformAPIs(t *testing.T) {
235 testJava(t, `
236 android_app {
237 name: "foo",
238 srcs: ["a.java"],
239 platform_apis: true,
240 }
241 `)
242
243 testJava(t, `
244 android_app {
245 name: "foo",
246 srcs: ["a.java"],
247 sdk_version: "current",
248 }
249 `)
250
251 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
252 android_app {
253 name: "bar",
254 srcs: ["b.java"],
255 }
256 `)
257
258 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
259 android_app {
260 name: "bar",
261 srcs: ["b.java"],
262 sdk_version: "system_current",
263 platform_apis: true,
264 }
265 `)
266}
267
Jeongik Chae403e9e2019-12-07 00:16:24 +0900268func TestAndroidAppLinkType(t *testing.T) {
269 testJava(t, `
270 android_app {
271 name: "foo",
272 srcs: ["a.java"],
273 libs: ["bar"],
274 static_libs: ["baz"],
275 platform_apis: true,
276 }
277
278 java_library {
279 name: "bar",
280 sdk_version: "current",
281 srcs: ["b.java"],
282 }
283
284 android_library {
285 name: "baz",
286 sdk_version: "system_current",
287 srcs: ["c.java"],
288 }
289 `)
290
291 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.", `
292 android_app {
293 name: "foo",
294 srcs: ["a.java"],
295 libs: ["bar"],
296 sdk_version: "current",
297 static_libs: ["baz"],
298 }
299
300 java_library {
301 name: "bar",
302 sdk_version: "current",
303 srcs: ["b.java"],
304 }
305
306 android_library {
307 name: "baz",
308 sdk_version: "system_current",
309 srcs: ["c.java"],
310 }
311 `)
312
313 testJava(t, `
314 android_app {
315 name: "foo",
316 srcs: ["a.java"],
317 libs: ["bar"],
318 sdk_version: "system_current",
319 static_libs: ["baz"],
320 }
321
322 java_library {
323 name: "bar",
324 sdk_version: "current",
325 srcs: ["b.java"],
326 }
327
328 android_library {
329 name: "baz",
330 sdk_version: "system_current",
331 srcs: ["c.java"],
332 }
333 `)
334
335 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.", `
336 android_app {
337 name: "foo",
338 srcs: ["a.java"],
339 libs: ["bar"],
340 sdk_version: "system_current",
341 static_libs: ["baz"],
342 }
343
344 java_library {
345 name: "bar",
346 sdk_version: "current",
347 srcs: ["b.java"],
348 }
349
350 android_library {
351 name: "baz",
352 srcs: ["c.java"],
353 }
354 `)
355}
356
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100357func TestUpdatableApps(t *testing.T) {
358 testCases := []struct {
359 name string
360 bp string
361 expectedError string
362 }{
363 {
364 name: "Stable public SDK",
365 bp: `android_app {
366 name: "foo",
367 srcs: ["a.java"],
368 sdk_version: "29",
Artur Satayev11962102020-04-16 13:43:02 +0100369 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100370 updatable: true,
371 }`,
372 },
373 {
374 name: "Stable system SDK",
375 bp: `android_app {
376 name: "foo",
377 srcs: ["a.java"],
378 sdk_version: "system_29",
Artur Satayev11962102020-04-16 13:43:02 +0100379 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100380 updatable: true,
381 }`,
382 },
383 {
384 name: "Current public SDK",
385 bp: `android_app {
386 name: "foo",
387 srcs: ["a.java"],
388 sdk_version: "current",
Artur Satayev11962102020-04-16 13:43:02 +0100389 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100390 updatable: true,
391 }`,
392 },
393 {
394 name: "Current system SDK",
395 bp: `android_app {
396 name: "foo",
397 srcs: ["a.java"],
398 sdk_version: "system_current",
Artur Satayev11962102020-04-16 13:43:02 +0100399 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100400 updatable: true,
401 }`,
402 },
403 {
404 name: "Current module SDK",
405 bp: `android_app {
406 name: "foo",
407 srcs: ["a.java"],
408 sdk_version: "module_current",
Artur Satayev11962102020-04-16 13:43:02 +0100409 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100410 updatable: true,
411 }`,
412 },
413 {
414 name: "Current core SDK",
415 bp: `android_app {
416 name: "foo",
417 srcs: ["a.java"],
418 sdk_version: "core_current",
Artur Satayev11962102020-04-16 13:43:02 +0100419 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100420 updatable: true,
421 }`,
422 },
423 {
424 name: "No Platform APIs",
425 bp: `android_app {
426 name: "foo",
427 srcs: ["a.java"],
428 platform_apis: true,
Artur Satayev11962102020-04-16 13:43:02 +0100429 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100430 updatable: true,
431 }`,
432 expectedError: "Updatable apps must use stable SDKs",
433 },
434 {
435 name: "No Core Platform APIs",
436 bp: `android_app {
437 name: "foo",
438 srcs: ["a.java"],
439 sdk_version: "core_platform",
Artur Satayev11962102020-04-16 13:43:02 +0100440 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100441 updatable: true,
442 }`,
443 expectedError: "Updatable apps must use stable SDKs",
444 },
445 {
446 name: "No unspecified APIs",
447 bp: `android_app {
448 name: "foo",
449 srcs: ["a.java"],
450 updatable: true,
Artur Satayev11962102020-04-16 13:43:02 +0100451 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100452 }`,
453 expectedError: "Updatable apps must use stable SDK",
454 },
Artur Satayev11962102020-04-16 13:43:02 +0100455 {
456 name: "Must specify min_sdk_version",
457 bp: `android_app {
458 name: "app_without_min_sdk_version",
459 srcs: ["a.java"],
460 sdk_version: "29",
461 updatable: true,
462 }`,
463 expectedError: "updatable apps must set min_sdk_version.",
464 },
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100465 }
466
467 for _, test := range testCases {
468 t.Run(test.name, func(t *testing.T) {
469 if test.expectedError == "" {
470 testJava(t, test.bp)
471 } else {
472 testJavaError(t, test.expectedError, test.bp)
473 }
474 })
475 }
476}
477
Jooyung Hanaf7f91f2020-04-29 14:01:06 +0900478func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
479 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
480 android_app {
481 name: "foo",
482 srcs: ["a.java"],
483 updatable: true,
484 sdk_version: "current",
485 min_sdk_version: "current",
486 jni_libs: ["libjni"],
487 }
488
489 cc_library {
490 name: "libjni",
491 stl: "none",
492 system_shared_libs: [],
493 sdk_version: "current",
494 }
495 `)
496}
497
498func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
499 bp := cc.GatherRequiredDepsForTest(android.Android) + `
500 android_app {
501 name: "foo",
502 srcs: ["a.java"],
503 updatable: true,
504 sdk_version: "current",
505 min_sdk_version: "29",
506 jni_libs: ["libjni"],
507 }
508
509 cc_library {
510 name: "libjni",
511 stl: "none",
512 system_shared_libs: [],
513 sdk_version: "29",
514 }
515
516 ndk_prebuilt_object {
517 name: "ndk_crtbegin_so.29",
518 sdk_version: "29",
519 }
520
521 ndk_prebuilt_object {
522 name: "ndk_crtend_so.29",
523 sdk_version: "29",
524 }
525 `
526 fs := map[string][]byte{
527 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
528 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
529 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
530 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
531 }
532
533 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
534
535 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
536 var crtbeginFound, crtendFound bool
537 for _, input := range inputs {
538 switch input.String() {
539 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o":
540 crtbeginFound = true
541 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o":
542 crtendFound = true
543 }
544 }
545 if !crtbeginFound || !crtendFound {
546 t.Error("should link with ndk_crtbegin_so.29 and ndk_crtend_so.29")
547 }
548}
549
550func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
551 bp := cc.GatherRequiredDepsForTest(android.Android) + `
552 android_app {
553 name: "foo",
554 srcs: ["a.java"],
555 updatable: true,
556 sdk_version: "current",
557 min_sdk_version: "29", // this APK should support 29
558 jni_libs: ["libjni"],
559 }
560
561 cc_library {
562 name: "libjni",
563 stl: "none",
564 sdk_version: "current",
565 }
566 `
567 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
568}
569
570func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
571 bp := cc.GatherRequiredDepsForTest(android.Android) + `
572 android_app {
573 name: "foo",
574 srcs: ["a.java"],
575 updatable: true,
576 sdk_version: "current",
577 min_sdk_version: "29", // this APK should support 29
578 jni_libs: ["libjni"],
579 }
580
581 cc_library {
582 name: "libjni",
583 stl: "none",
584 shared_libs: ["libbar"],
585 system_shared_libs: [],
586 sdk_version: "27",
587 }
588
589 cc_library {
590 name: "libbar",
591 stl: "none",
592 system_shared_libs: [],
593 sdk_version: "current",
594 }
595 `
596 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
597}
598
Colin Cross0ddae7f2019-02-07 15:30:01 -0800599func TestResourceDirs(t *testing.T) {
600 testCases := []struct {
601 name string
602 prop string
603 resources []string
604 }{
605 {
606 name: "no resource_dirs",
607 prop: "",
608 resources: []string{"res/res/values/strings.xml"},
609 },
610 {
611 name: "resource_dirs",
612 prop: `resource_dirs: ["res"]`,
613 resources: []string{"res/res/values/strings.xml"},
614 },
615 {
616 name: "empty resource_dirs",
617 prop: `resource_dirs: []`,
618 resources: nil,
619 },
620 }
621
622 fs := map[string][]byte{
623 "res/res/values/strings.xml": nil,
624 }
625
626 bp := `
627 android_app {
628 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900629 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800630 %s
631 }
632 `
633
634 for _, testCase := range testCases {
635 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800636 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
637 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800638 run(t, ctx, config)
639
640 module := ctx.ModuleForTests("foo", "android_common")
641 resourceList := module.MaybeOutput("aapt2/res.list")
642
643 var resources []string
644 if resourceList.Rule != nil {
645 for _, compiledResource := range resourceList.Inputs.Strings() {
646 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
647 }
648 }
649
650 if !reflect.DeepEqual(resources, testCase.resources) {
651 t.Errorf("expected resource files %q, got %q",
652 testCase.resources, resources)
653 }
654 })
655 }
656}
657
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800658func TestLibraryAssets(t *testing.T) {
659 bp := `
660 android_app {
661 name: "foo",
662 sdk_version: "current",
663 static_libs: ["lib1", "lib2", "lib3"],
664 }
665
666 android_library {
667 name: "lib1",
668 sdk_version: "current",
669 asset_dirs: ["assets_a"],
670 }
671
672 android_library {
673 name: "lib2",
674 sdk_version: "current",
675 }
676
677 android_library {
678 name: "lib3",
679 sdk_version: "current",
680 static_libs: ["lib4"],
681 }
682
683 android_library {
684 name: "lib4",
685 sdk_version: "current",
686 asset_dirs: ["assets_b"],
687 }
688 `
689
690 testCases := []struct {
691 name string
692 assetFlag string
693 assetPackages []string
694 }{
695 {
696 name: "foo",
697 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
698 assetPackages: []string{
699 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
700 buildDir + "/.intermediates/lib1/android_common/assets.zip",
701 buildDir + "/.intermediates/lib3/android_common/assets.zip",
702 },
703 },
704 {
705 name: "lib1",
706 assetFlag: "-A assets_a",
707 },
708 {
709 name: "lib2",
710 },
711 {
712 name: "lib3",
713 assetPackages: []string{
714 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
715 buildDir + "/.intermediates/lib4/android_common/assets.zip",
716 },
717 },
718 {
719 name: "lib4",
720 assetFlag: "-A assets_b",
721 },
722 }
723 ctx := testApp(t, bp)
724
725 for _, test := range testCases {
726 t.Run(test.name, func(t *testing.T) {
727 m := ctx.ModuleForTests(test.name, "android_common")
728
729 // Check asset flag in aapt2 link flags
730 var aapt2link android.TestingBuildParams
731 if len(test.assetPackages) > 0 {
732 aapt2link = m.Output("aapt2/package-res.apk")
733 } else {
734 aapt2link = m.Output("package-res.apk")
735 }
736 aapt2Flags := aapt2link.Args["flags"]
737 if test.assetFlag != "" {
738 if !strings.Contains(aapt2Flags, test.assetFlag) {
739 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
740 }
741 } else {
742 if strings.Contains(aapt2Flags, " -A ") {
743 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
744 }
745 }
746
747 // Check asset merge rule.
748 if len(test.assetPackages) > 0 {
749 mergeAssets := m.Output("package-res.apk")
750 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
751 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
752 mergeAssets.Inputs.Strings(), test.assetPackages)
753 }
754 }
755 })
756 }
757}
758
Colin Crossbec85302019-02-13 13:15:46 -0800759func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800760 testCases := []struct {
761 name string
762 enforceRROTargets []string
763 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800764 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800765 overlayFiles map[string][]string
766 rroDirs map[string][]string
767 }{
768 {
769 name: "no RRO",
770 enforceRROTargets: nil,
771 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800772 resourceFiles: map[string][]string{
773 "foo": nil,
774 "bar": {"bar/res/res/values/strings.xml"},
775 "lib": nil,
776 "lib2": {"lib2/res/res/values/strings.xml"},
777 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800778 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800779 "foo": {
780 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800781 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000782 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800783 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800784 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
785 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000786 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800787 },
Colin Crossbec85302019-02-13 13:15:46 -0800788 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800789 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
790 "device/vendor/blah/overlay/bar/res/values/strings.xml",
791 },
Colin Crossbec85302019-02-13 13:15:46 -0800792 "lib": {
793 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
794 "lib/res/res/values/strings.xml",
795 "device/vendor/blah/overlay/lib/res/values/strings.xml",
796 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800797 },
798 rroDirs: map[string][]string{
799 "foo": nil,
800 "bar": nil,
801 },
802 },
803 {
804 name: "enforce RRO on foo",
805 enforceRROTargets: []string{"foo"},
806 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800807 resourceFiles: map[string][]string{
808 "foo": nil,
809 "bar": {"bar/res/res/values/strings.xml"},
810 "lib": nil,
811 "lib2": {"lib2/res/res/values/strings.xml"},
812 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800813 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800814 "foo": {
815 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800816 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000817 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800818 "foo/res/res/values/strings.xml",
819 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
820 },
Colin Crossbec85302019-02-13 13:15:46 -0800821 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800822 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
823 "device/vendor/blah/overlay/bar/res/values/strings.xml",
824 },
Colin Crossbec85302019-02-13 13:15:46 -0800825 "lib": {
826 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
827 "lib/res/res/values/strings.xml",
828 "device/vendor/blah/overlay/lib/res/values/strings.xml",
829 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800830 },
Colin Crossc1c37552019-01-31 11:42:41 -0800831
Colin Cross5c4791c2019-02-01 11:44:44 -0800832 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800833 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000834 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800835 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
836 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000837 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800838 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800839 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800840 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800841 },
842 },
843 {
844 name: "enforce RRO on all",
845 enforceRROTargets: []string{"*"},
846 enforceRROExcludedOverlays: []string{
847 // Excluding specific apps/res directories also allowed.
848 "device/vendor/blah/static_overlay/foo",
849 "device/vendor/blah/static_overlay/bar/res",
850 },
Colin Crossbec85302019-02-13 13:15:46 -0800851 resourceFiles: map[string][]string{
852 "foo": nil,
853 "bar": {"bar/res/res/values/strings.xml"},
854 "lib": nil,
855 "lib2": {"lib2/res/res/values/strings.xml"},
856 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800857 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800858 "foo": {
859 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800860 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000861 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800862 "foo/res/res/values/strings.xml",
863 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
864 },
Colin Crossbec85302019-02-13 13:15:46 -0800865 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
866 "lib": {
867 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
868 "lib/res/res/values/strings.xml",
869 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800870 },
871 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800872 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000873 "device:device/vendor/blah/overlay/foo/res",
874 "product:product/vendor/blah/overlay/foo/res",
875 // Lib dep comes after the direct deps
876 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800877 },
Anton Hansson53c88442019-03-18 15:53:16 +0000878 "bar": {"device:device/vendor/blah/overlay/bar/res"},
879 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800880 },
881 },
882 }
883
Anton Hansson53c88442019-03-18 15:53:16 +0000884 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800885 "device/vendor/blah/overlay",
886 "device/vendor/blah/overlay2",
887 "device/vendor/blah/static_overlay",
888 }
889
Anton Hansson53c88442019-03-18 15:53:16 +0000890 productResourceOverlays := []string{
891 "product/vendor/blah/overlay",
892 }
893
Colin Cross890ff552017-11-30 20:13:19 -0800894 fs := map[string][]byte{
895 "foo/res/res/values/strings.xml": nil,
896 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800897 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800898 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800899 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
900 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800901 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800902 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
903 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
904 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000905 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800906 }
907
908 bp := `
909 android_app {
910 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900911 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800912 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000913 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800914 }
915
916 android_app {
917 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900918 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800919 resource_dirs: ["bar/res"],
920 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800921
922 android_library {
923 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900924 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800925 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800926 static_libs: ["lib2"],
927 }
928
929 android_library {
930 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900931 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800932 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800933 }
Anton Hansson53c88442019-03-18 15:53:16 +0000934
935 // This library has the same resources as lib (should not lead to dupe RROs)
936 android_library {
937 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900938 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000939 resource_dirs: ["lib/res"]
940 }
Colin Cross890ff552017-11-30 20:13:19 -0800941 `
942
Colin Cross5c4791c2019-02-01 11:44:44 -0800943 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800944 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800945 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000946 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
947 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800948 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800949 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800950 }
951 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800952 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800953 }
954
Colin Cross98be1bb2019-12-13 20:41:13 -0800955 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800956 run(t, ctx, config)
957
Colin Crossbec85302019-02-13 13:15:46 -0800958 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
959 for _, o := range list {
960 res := module.MaybeOutput(o)
961 if res.Rule != nil {
962 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
963 // verify the inputs to the .arsc.flat rule.
964 files = append(files, res.Inputs.Strings()...)
965 } else {
966 // Otherwise, verify the full path to the output of the other module
967 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000968 }
Colin Cross890ff552017-11-30 20:13:19 -0800969 }
Colin Crossbec85302019-02-13 13:15:46 -0800970 return files
Colin Cross890ff552017-11-30 20:13:19 -0800971 }
972
Colin Crossbec85302019-02-13 13:15:46 -0800973 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
974 module := ctx.ModuleForTests(moduleName, "android_common")
975 resourceList := module.MaybeOutput("aapt2/res.list")
976 if resourceList.Rule != nil {
977 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000978 }
Colin Crossbec85302019-02-13 13:15:46 -0800979 overlayList := module.MaybeOutput("aapt2/overlay.list")
980 if overlayList.Rule != nil {
981 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
982 }
983
Anton Hansson53c88442019-03-18 15:53:16 +0000984 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
985 var prefix string
986 if d.overlayType == device {
987 prefix = "device:"
988 } else if d.overlayType == product {
989 prefix = "product:"
990 } else {
991 t.Fatalf("Unexpected overlayType %d", d.overlayType)
992 }
993 rroDirs = append(rroDirs, prefix+d.path.String())
994 }
Colin Crossbec85302019-02-13 13:15:46 -0800995
996 return resourceFiles, overlayFiles, rroDirs
997 }
998
999 modules := []string{"foo", "bar", "lib", "lib2"}
1000 for _, module := range modules {
1001 resourceFiles, overlayFiles, rroDirs := getResources(module)
1002
1003 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
1004 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
1005 module, testCase.resourceFiles[module], resourceFiles)
1006 }
1007 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
1008 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
1009 module, testCase.overlayFiles[module], overlayFiles)
1010 }
1011 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +00001012 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -08001013 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +00001014 }
Colin Cross890ff552017-11-30 20:13:19 -08001015 }
Colin Cross890ff552017-11-30 20:13:19 -08001016 })
1017 }
1018}
Colin Crossd09b0b62018-04-18 11:06:47 -07001019
1020func TestAppSdkVersion(t *testing.T) {
1021 testCases := []struct {
1022 name string
1023 sdkVersion string
1024 platformSdkInt int
1025 platformSdkCodename string
1026 platformSdkFinal bool
1027 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +09001028 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -07001029 }{
1030 {
1031 name: "current final SDK",
1032 sdkVersion: "current",
1033 platformSdkInt: 27,
1034 platformSdkCodename: "REL",
1035 platformSdkFinal: true,
1036 expectedMinSdkVersion: "27",
1037 },
1038 {
1039 name: "current non-final SDK",
1040 sdkVersion: "current",
1041 platformSdkInt: 27,
1042 platformSdkCodename: "OMR1",
1043 platformSdkFinal: false,
1044 expectedMinSdkVersion: "OMR1",
1045 },
1046 {
1047 name: "default final SDK",
1048 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001049 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001050 platformSdkInt: 27,
1051 platformSdkCodename: "REL",
1052 platformSdkFinal: true,
1053 expectedMinSdkVersion: "27",
1054 },
1055 {
1056 name: "default non-final SDK",
1057 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001058 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001059 platformSdkInt: 27,
1060 platformSdkCodename: "OMR1",
1061 platformSdkFinal: false,
1062 expectedMinSdkVersion: "OMR1",
1063 },
1064 {
1065 name: "14",
1066 sdkVersion: "14",
1067 expectedMinSdkVersion: "14",
1068 },
1069 }
1070
1071 for _, moduleType := range []string{"android_app", "android_library"} {
1072 for _, test := range testCases {
1073 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001074 platformApiProp := ""
1075 if test.platformApis {
1076 platformApiProp = "platform_apis: true,"
1077 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001078 bp := fmt.Sprintf(`%s {
1079 name: "foo",
1080 srcs: ["a.java"],
1081 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001082 %s
1083 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001084
Colin Cross98be1bb2019-12-13 20:41:13 -08001085 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001086 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1087 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1088 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1089
Colin Cross98be1bb2019-12-13 20:41:13 -08001090 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -07001091
1092 run(t, ctx, config)
1093
1094 foo := ctx.ModuleForTests("foo", "android_common")
1095 link := foo.Output("package-res.apk")
1096 linkFlags := strings.Split(link.Args["flags"], " ")
1097 min := android.IndexList("--min-sdk-version", linkFlags)
1098 target := android.IndexList("--target-sdk-version", linkFlags)
1099
1100 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1101 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1102 }
1103
1104 gotMinSdkVersion := linkFlags[min+1]
1105 gotTargetSdkVersion := linkFlags[target+1]
1106
1107 if gotMinSdkVersion != test.expectedMinSdkVersion {
1108 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1109 test.expectedMinSdkVersion, gotMinSdkVersion)
1110 }
1111
1112 if gotTargetSdkVersion != test.expectedMinSdkVersion {
1113 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1114 test.expectedMinSdkVersion, gotTargetSdkVersion)
1115 }
1116 })
1117 }
1118 }
1119}
Colin Crossa4f08812018-10-02 22:03:40 -07001120
Paul Duffin50c217c2019-06-12 13:25:22 +01001121func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001122 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001123 cc_library {
1124 name: "libjni",
1125 system_shared_libs: [],
Colin Cross01fd7cc2020-02-19 16:54:04 -08001126 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001127 stl: "none",
1128 }
1129
1130 android_test {
1131 name: "test",
1132 sdk_version: "core_platform",
1133 jni_libs: ["libjni"],
1134 }
1135
1136 android_test {
1137 name: "test_first",
1138 sdk_version: "core_platform",
1139 compile_multilib: "first",
1140 jni_libs: ["libjni"],
1141 }
1142
1143 android_test {
1144 name: "test_both",
1145 sdk_version: "core_platform",
1146 compile_multilib: "both",
1147 jni_libs: ["libjni"],
1148 }
1149
1150 android_test {
1151 name: "test_32",
1152 sdk_version: "core_platform",
1153 compile_multilib: "32",
1154 jni_libs: ["libjni"],
1155 }
1156
1157 android_test {
1158 name: "test_64",
1159 sdk_version: "core_platform",
1160 compile_multilib: "64",
1161 jni_libs: ["libjni"],
1162 }
1163 `)
1164
1165 testCases := []struct {
1166 name string
1167 abis []string
1168 }{
1169 {"test", []string{"arm64-v8a"}},
1170 {"test_first", []string{"arm64-v8a"}},
1171 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1172 {"test_32", []string{"armeabi-v7a"}},
1173 {"test_64", []string{"arm64-v8a"}},
1174 }
1175
1176 for _, test := range testCases {
1177 t.Run(test.name, func(t *testing.T) {
1178 app := ctx.ModuleForTests(test.name, "android_common")
1179 jniLibZip := app.Output("jnilibs.zip")
1180 var abis []string
1181 args := strings.Fields(jniLibZip.Args["jarArgs"])
1182 for i := 0; i < len(args); i++ {
1183 if args[i] == "-P" {
1184 abis = append(abis, filepath.Base(args[i+1]))
1185 i++
1186 }
1187 }
1188 if !reflect.DeepEqual(abis, test.abis) {
1189 t.Errorf("want abis %v, got %v", test.abis, abis)
1190 }
1191 })
1192 }
1193}
1194
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001195func TestAppSdkVersionByPartition(t *testing.T) {
1196 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1197 android_app {
1198 name: "foo",
1199 srcs: ["a.java"],
1200 vendor: true,
1201 platform_apis: true,
1202 }
1203 `)
1204
1205 testJava(t, `
1206 android_app {
1207 name: "bar",
1208 srcs: ["b.java"],
1209 platform_apis: true,
1210 }
1211 `)
1212
1213 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001214 bp := `
1215 android_app {
1216 name: "foo",
1217 srcs: ["a.java"],
1218 product_specific: true,
1219 platform_apis: true,
1220 }
1221 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001222
1223 config := testAppConfig(nil, bp, nil)
1224 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001225 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001226 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 +09001227 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001228 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001229 }
1230 }
1231}
1232
Paul Duffin50c217c2019-06-12 13:25:22 +01001233func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001234 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001235 cc_library {
1236 name: "libjni",
1237 system_shared_libs: [],
1238 stl: "none",
Colin Cross1c93c292020-02-15 10:38:00 -08001239 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001240 }
1241
1242 android_app {
1243 name: "app",
1244 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001245 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001246 }
1247
1248 android_app {
1249 name: "app_noembed",
1250 jni_libs: ["libjni"],
1251 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001252 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001253 }
1254
1255 android_app {
1256 name: "app_embed",
1257 jni_libs: ["libjni"],
1258 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001259 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001260 }
1261
1262 android_test {
1263 name: "test",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001264 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001265 jni_libs: ["libjni"],
1266 }
1267
1268 android_test {
1269 name: "test_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001270 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001271 jni_libs: ["libjni"],
1272 use_embedded_native_libs: false,
1273 }
1274
1275 android_test_helper_app {
1276 name: "test_helper",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001277 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001278 jni_libs: ["libjni"],
1279 }
1280
1281 android_test_helper_app {
1282 name: "test_helper_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001283 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001284 jni_libs: ["libjni"],
1285 use_embedded_native_libs: false,
1286 }
1287 `)
1288
1289 testCases := []struct {
1290 name string
1291 packaged bool
1292 compressed bool
1293 }{
1294 {"app", false, false},
1295 {"app_noembed", false, false},
1296 {"app_embed", true, false},
1297 {"test", true, false},
1298 {"test_noembed", true, true},
1299 {"test_helper", true, false},
1300 {"test_helper_noembed", true, true},
1301 }
1302
1303 for _, test := range testCases {
1304 t.Run(test.name, func(t *testing.T) {
1305 app := ctx.ModuleForTests(test.name, "android_common")
1306 jniLibZip := app.MaybeOutput("jnilibs.zip")
1307 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1308 t.Errorf("expected jni packaged %v, got %v", w, g)
1309 }
1310
1311 if jniLibZip.Rule != nil {
1312 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1313 t.Errorf("expected jni compressed %v, got %v", w, g)
1314 }
Colin Cross01fd7cc2020-02-19 16:54:04 -08001315
1316 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1317 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1318 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001319 }
1320 })
1321 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001322}
1323
Colin Cross1dd9c442020-05-08 11:20:24 -07001324func TestJNISDK(t *testing.T) {
1325 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1326 cc_library {
1327 name: "libjni",
1328 system_shared_libs: [],
1329 stl: "none",
1330 sdk_version: "current",
1331 }
1332
1333 android_test {
1334 name: "app_platform",
1335 jni_libs: ["libjni"],
1336 platform_apis: true,
1337 }
1338
1339 android_test {
1340 name: "app_sdk",
1341 jni_libs: ["libjni"],
1342 sdk_version: "current",
1343 }
1344
1345 android_test {
1346 name: "app_force_platform",
1347 jni_libs: ["libjni"],
1348 sdk_version: "current",
1349 jni_uses_platform_apis: true,
1350 }
1351
1352 android_test {
1353 name: "app_force_sdk",
1354 jni_libs: ["libjni"],
1355 platform_apis: true,
1356 jni_uses_sdk_apis: true,
1357 }
Colin Crosseb032962020-05-13 11:05:02 -07001358
1359 cc_library {
1360 name: "libvendorjni",
1361 system_shared_libs: [],
1362 stl: "none",
1363 vendor: true,
1364 }
1365
1366 android_test {
1367 name: "app_vendor",
1368 jni_libs: ["libvendorjni"],
1369 sdk_version: "current",
1370 vendor: true,
1371 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001372 `)
1373
1374 testCases := []struct {
Colin Crosseb032962020-05-13 11:05:02 -07001375 name string
1376 sdkJNI bool
1377 vendorJNI bool
Colin Cross1dd9c442020-05-08 11:20:24 -07001378 }{
Colin Crosseb032962020-05-13 11:05:02 -07001379 {name: "app_platform"},
1380 {name: "app_sdk", sdkJNI: true},
1381 {name: "app_force_platform"},
1382 {name: "app_force_sdk", sdkJNI: true},
1383 {name: "app_vendor", vendorJNI: true},
Colin Cross1dd9c442020-05-08 11:20:24 -07001384 }
1385
Colin Crosseb032962020-05-13 11:05:02 -07001386 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1387 Output("libjni.so").Output.String()
1388 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1389 Output("libjni.so").Output.String()
1390 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1391 Output("libvendorjni.so").Output.String()
1392
Colin Cross1dd9c442020-05-08 11:20:24 -07001393 for _, test := range testCases {
1394 t.Run(test.name, func(t *testing.T) {
1395 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross1dd9c442020-05-08 11:20:24 -07001396
1397 jniLibZip := app.MaybeOutput("jnilibs.zip")
1398 if len(jniLibZip.Implicits) != 1 {
1399 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1400 }
1401 gotJNI := jniLibZip.Implicits[0].String()
1402
1403 if test.sdkJNI {
1404 if gotJNI != sdkJNI {
1405 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1406 }
Colin Crosseb032962020-05-13 11:05:02 -07001407 } else if test.vendorJNI {
1408 if gotJNI != vendorJNI {
1409 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1410 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001411 } else {
1412 if gotJNI != platformJNI {
1413 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1414 }
1415 }
1416 })
1417 }
1418
1419 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1420 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1421 android_test {
1422 name: "app_platform",
1423 platform_apis: true,
1424 jni_uses_platform_apis: true,
1425 }
1426 `)
1427 })
1428
1429 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1430 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1431 android_test {
1432 name: "app_sdk",
1433 sdk_version: "current",
1434 jni_uses_sdk_apis: true,
1435 }
1436 `)
1437 })
1438
1439}
1440
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001441func TestCertificates(t *testing.T) {
1442 testCases := []struct {
1443 name string
1444 bp string
1445 certificateOverride string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001446 expectedLineage string
1447 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001448 }{
1449 {
1450 name: "default",
1451 bp: `
1452 android_app {
1453 name: "foo",
1454 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001455 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001456 }
1457 `,
1458 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001459 expectedLineage: "",
1460 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001461 },
1462 {
1463 name: "module certificate property",
1464 bp: `
1465 android_app {
1466 name: "foo",
1467 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001468 certificate: ":new_certificate",
1469 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001470 }
1471
1472 android_app_certificate {
1473 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001474 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001475 }
1476 `,
1477 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001478 expectedLineage: "",
1479 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001480 },
1481 {
1482 name: "path certificate property",
1483 bp: `
1484 android_app {
1485 name: "foo",
1486 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001487 certificate: "expiredkey",
1488 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001489 }
1490 `,
1491 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001492 expectedLineage: "",
1493 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001494 },
1495 {
1496 name: "certificate overrides",
1497 bp: `
1498 android_app {
1499 name: "foo",
1500 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001501 certificate: "expiredkey",
1502 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001503 }
1504
1505 android_app_certificate {
1506 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001507 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001508 }
1509 `,
1510 certificateOverride: "foo:new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001511 expectedLineage: "",
1512 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1513 },
1514 {
1515 name: "certificate lineage",
1516 bp: `
1517 android_app {
1518 name: "foo",
1519 srcs: ["a.java"],
1520 certificate: ":new_certificate",
1521 lineage: "lineage.bin",
1522 sdk_version: "current",
1523 }
1524
1525 android_app_certificate {
1526 name: "new_certificate",
1527 certificate: "cert/new_cert",
1528 }
1529 `,
1530 certificateOverride: "",
1531 expectedLineage: "--lineage lineage.bin",
1532 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001533 },
1534 }
1535
1536 for _, test := range testCases {
1537 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001538 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001539 if test.certificateOverride != "" {
1540 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1541 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001542 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001543
1544 run(t, ctx, config)
1545 foo := ctx.ModuleForTests("foo", "android_common")
1546
1547 signapk := foo.Output("foo.apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001548 signCertificateFlags := signapk.Args["certificates"]
1549 if test.expectedCertificate != signCertificateFlags {
1550 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1551 }
1552
1553 signFlags := signapk.Args["flags"]
1554 if test.expectedLineage != signFlags {
1555 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001556 }
1557 })
1558 }
1559}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001560
Songchun Fan688de9a2020-03-24 20:32:24 -07001561func TestRequestV4SigningFlag(t *testing.T) {
1562 testCases := []struct {
1563 name string
1564 bp string
1565 expected string
1566 }{
1567 {
1568 name: "default",
1569 bp: `
1570 android_app {
1571 name: "foo",
1572 srcs: ["a.java"],
1573 sdk_version: "current",
1574 }
1575 `,
1576 expected: "",
1577 },
1578 {
1579 name: "default",
1580 bp: `
1581 android_app {
1582 name: "foo",
1583 srcs: ["a.java"],
1584 sdk_version: "current",
1585 v4_signature: false,
1586 }
1587 `,
1588 expected: "",
1589 },
1590 {
1591 name: "module certificate property",
1592 bp: `
1593 android_app {
1594 name: "foo",
1595 srcs: ["a.java"],
1596 sdk_version: "current",
1597 v4_signature: true,
1598 }
1599 `,
1600 expected: "--enable-v4",
1601 },
1602 }
1603
1604 for _, test := range testCases {
1605 t.Run(test.name, func(t *testing.T) {
1606 config := testAppConfig(nil, test.bp, nil)
1607 ctx := testContext()
1608
1609 run(t, ctx, config)
1610 foo := ctx.ModuleForTests("foo", "android_common")
1611
1612 signapk := foo.Output("foo.apk")
1613 signFlags := signapk.Args["flags"]
1614 if test.expected != signFlags {
1615 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1616 }
1617 })
1618 }
1619}
1620
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001621func TestPackageNameOverride(t *testing.T) {
1622 testCases := []struct {
1623 name string
1624 bp string
1625 packageNameOverride string
1626 expected []string
1627 }{
1628 {
1629 name: "default",
1630 bp: `
1631 android_app {
1632 name: "foo",
1633 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001634 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001635 }
1636 `,
1637 packageNameOverride: "",
1638 expected: []string{
1639 buildDir + "/.intermediates/foo/android_common/foo.apk",
1640 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1641 },
1642 },
1643 {
1644 name: "overridden",
1645 bp: `
1646 android_app {
1647 name: "foo",
1648 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001649 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001650 }
1651 `,
1652 packageNameOverride: "foo:bar",
1653 expected: []string{
1654 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001655 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001656 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1657 },
1658 },
1659 }
1660
1661 for _, test := range testCases {
1662 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001663 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001664 if test.packageNameOverride != "" {
1665 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1666 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001667 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001668
1669 run(t, ctx, config)
1670 foo := ctx.ModuleForTests("foo", "android_common")
1671
1672 outputs := foo.AllOutputs()
1673 outputMap := make(map[string]bool)
1674 for _, o := range outputs {
1675 outputMap[o] = true
1676 }
1677 for _, e := range test.expected {
1678 if _, exist := outputMap[e]; !exist {
1679 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1680 }
1681 }
1682 })
1683 }
1684}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001685
1686func TestInstrumentationTargetOverridden(t *testing.T) {
1687 bp := `
1688 android_app {
1689 name: "foo",
1690 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001691 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001692 }
1693
1694 android_test {
1695 name: "bar",
1696 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001697 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001698 }
1699 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001700 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001701 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001702 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001703
1704 run(t, ctx, config)
1705
1706 bar := ctx.ModuleForTests("bar", "android_common")
1707 res := bar.Output("package-res.apk")
1708 aapt2Flags := res.Args["flags"]
1709 e := "--rename-instrumentation-target-package org.dandroid.bp"
1710 if !strings.Contains(aapt2Flags, e) {
1711 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1712 }
1713}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001714
1715func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001716 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001717 android_app {
1718 name: "foo",
1719 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001720 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001721 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001722 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001723 }
1724
1725 override_android_app {
1726 name: "bar",
1727 base: "foo",
1728 certificate: ":new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001729 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001730 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001731 }
1732
1733 android_app_certificate {
1734 name: "new_certificate",
1735 certificate: "cert/new_cert",
1736 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001737
1738 override_android_app {
1739 name: "baz",
1740 base: "foo",
1741 package_name: "org.dandroid.bp",
1742 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001743 `)
1744
1745 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001746 moduleName string
1747 variantName string
1748 apkName string
1749 apkPath string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001750 certFlag string
1751 lineageFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001752 overrides []string
1753 aaptFlag string
1754 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001755 }{
1756 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001757 moduleName: "foo",
1758 variantName: "android_common",
1759 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001760 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1761 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001762 overrides: []string{"qux"},
1763 aaptFlag: "",
1764 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001765 },
1766 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001767 moduleName: "bar",
1768 variantName: "android_common_bar",
1769 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001770 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1771 lineageFlag: "--lineage lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001772 overrides: []string{"qux", "foo"},
1773 aaptFlag: "",
1774 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001775 },
1776 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001777 moduleName: "baz",
1778 variantName: "android_common_baz",
1779 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001780 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1781 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001782 overrides: []string{"qux", "foo"},
1783 aaptFlag: "--rename-manifest-package org.dandroid.bp",
1784 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001785 },
1786 }
1787 for _, expected := range expectedVariants {
1788 variant := ctx.ModuleForTests("foo", expected.variantName)
1789
1790 // Check the final apk name
1791 outputs := variant.AllOutputs()
1792 expectedApkPath := buildDir + expected.apkPath
1793 found := false
1794 for _, o := range outputs {
1795 if o == expectedApkPath {
1796 found = true
1797 break
1798 }
1799 }
1800 if !found {
1801 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1802 }
1803
1804 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001805 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001806 certFlag := signapk.Args["certificates"]
1807 if expected.certFlag != certFlag {
1808 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1809 }
1810
1811 // Check the lineage flags
1812 lineageFlag := signapk.Args["flags"]
1813 if expected.lineageFlag != lineageFlag {
1814 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001815 }
1816
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001817 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001818 mod := variant.Module().(*AndroidApp)
1819 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1820 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1821 expected.overrides, mod.appProperties.Overrides)
1822 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001823
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001824 // Test Overridable property: Logging_parent
1825 logging_parent := mod.aapt.LoggingParent
1826 if expected.logging_parent != logging_parent {
1827 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1828 expected.logging_parent, logging_parent)
1829 }
1830
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001831 // Check the package renaming flag, if exists.
1832 res := variant.Output("package-res.apk")
1833 aapt2Flags := res.Args["flags"]
1834 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1835 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1836 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001837 }
1838}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001839
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001840func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001841 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001842 android_app {
1843 name: "foo",
1844 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001845 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001846 }
1847
1848 override_android_app {
1849 name: "bar",
1850 base: "foo",
1851 package_name: "org.dandroid.bp",
1852 }
1853
1854 android_test {
1855 name: "baz",
1856 srcs: ["b.java"],
1857 instrumentation_for: "foo",
1858 }
1859
1860 android_test {
1861 name: "qux",
1862 srcs: ["b.java"],
1863 instrumentation_for: "bar",
1864 }
1865 `)
1866
1867 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1868 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1869 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1870 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1871 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1872 }
1873
1874 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1875 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1876 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1877 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1878 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1879 }
1880}
1881
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001882func TestOverrideAndroidTest(t *testing.T) {
1883 ctx, _ := testJava(t, `
1884 android_app {
1885 name: "foo",
1886 srcs: ["a.java"],
1887 package_name: "com.android.foo",
1888 sdk_version: "current",
1889 }
1890
1891 override_android_app {
1892 name: "bar",
1893 base: "foo",
1894 package_name: "com.android.bar",
1895 }
1896
1897 android_test {
1898 name: "foo_test",
1899 srcs: ["b.java"],
1900 instrumentation_for: "foo",
1901 }
1902
1903 override_android_test {
1904 name: "bar_test",
1905 base: "foo_test",
1906 package_name: "com.android.bar.test",
1907 instrumentation_for: "bar",
1908 instrumentation_target_package: "com.android.bar",
1909 }
1910 `)
1911
1912 expectedVariants := []struct {
1913 moduleName string
1914 variantName string
1915 apkPath string
1916 overrides []string
1917 targetVariant string
1918 packageFlag string
1919 targetPackageFlag string
1920 }{
1921 {
1922 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001923 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001924 overrides: nil,
1925 targetVariant: "android_common",
1926 packageFlag: "",
1927 targetPackageFlag: "",
1928 },
1929 {
1930 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001931 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001932 overrides: []string{"foo_test"},
1933 targetVariant: "android_common_bar",
1934 packageFlag: "com.android.bar.test",
1935 targetPackageFlag: "com.android.bar",
1936 },
1937 }
1938 for _, expected := range expectedVariants {
1939 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1940
1941 // Check the final apk name
1942 outputs := variant.AllOutputs()
1943 expectedApkPath := buildDir + expected.apkPath
1944 found := false
1945 for _, o := range outputs {
1946 if o == expectedApkPath {
1947 found = true
1948 break
1949 }
1950 }
1951 if !found {
1952 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1953 }
1954
1955 // Check if the overrides field values are correctly aggregated.
1956 mod := variant.Module().(*AndroidTest)
1957 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1958 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1959 expected.overrides, mod.appProperties.Overrides)
1960 }
1961
1962 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1963 javac := variant.Rule("javac")
1964 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1965 if !strings.Contains(javac.Args["classpath"], turbine) {
1966 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1967 }
1968
1969 // Check aapt2 flags.
1970 res := variant.Output("package-res.apk")
1971 aapt2Flags := res.Args["flags"]
1972 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1973 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1974 }
1975}
1976
Jaewoong Jung39982342020-01-14 10:27:18 -08001977func TestAndroidTest_FixTestConfig(t *testing.T) {
1978 ctx, _ := testJava(t, `
1979 android_app {
1980 name: "foo",
1981 srcs: ["a.java"],
1982 package_name: "com.android.foo",
1983 sdk_version: "current",
1984 }
1985
1986 android_test {
1987 name: "foo_test",
1988 srcs: ["b.java"],
1989 instrumentation_for: "foo",
1990 }
1991
1992 android_test {
1993 name: "bar_test",
1994 srcs: ["b.java"],
1995 package_name: "com.android.bar.test",
1996 instrumentation_for: "foo",
1997 }
1998
1999 override_android_test {
2000 name: "baz_test",
2001 base: "foo_test",
2002 package_name: "com.android.baz.test",
2003 }
2004 `)
2005
2006 testCases := []struct {
2007 moduleName string
2008 variantName string
2009 expectedFlags []string
2010 }{
2011 {
2012 moduleName: "foo_test",
2013 variantName: "android_common",
2014 },
2015 {
2016 moduleName: "bar_test",
2017 variantName: "android_common",
2018 expectedFlags: []string{
2019 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
2020 "--package-name com.android.bar.test",
2021 },
2022 },
2023 {
2024 moduleName: "foo_test",
2025 variantName: "android_common_baz_test",
2026 expectedFlags: []string{
2027 "--manifest " + buildDir +
2028 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
2029 "--package-name com.android.baz.test",
2030 "--test-file-name baz_test.apk",
2031 },
2032 },
2033 }
2034
2035 for _, test := range testCases {
2036 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
2037 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
2038
2039 if len(test.expectedFlags) > 0 {
2040 if params.Rule == nil {
2041 t.Errorf("test_config_fixer was expected to run, but didn't")
2042 } else {
2043 for _, flag := range test.expectedFlags {
2044 if !strings.Contains(params.RuleParams.Command, flag) {
2045 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2046 }
2047 }
2048 }
2049 } else {
2050 if params.Rule != nil {
2051 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2052 }
2053 }
2054
2055 }
2056}
2057
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002058func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002059 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002060 android_app_import {
2061 name: "foo",
2062 apk: "prebuilts/apk/app.apk",
2063 certificate: "platform",
2064 dex_preopt: {
2065 enabled: true,
2066 },
2067 }
2068 `)
2069
2070 variant := ctx.ModuleForTests("foo", "android_common")
2071
2072 // Check dexpreopt outputs.
2073 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2074 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2075 t.Errorf("can't find dexpreopt outputs")
2076 }
2077
2078 // Check cert signing flag.
2079 signedApk := variant.Output("signed/foo.apk")
2080 signingFlag := signedApk.Args["certificates"]
2081 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2082 if expected != signingFlag {
2083 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2084 }
2085}
2086
2087func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002088 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002089 android_app_import {
2090 name: "foo",
2091 apk: "prebuilts/apk/app.apk",
2092 certificate: "platform",
2093 dex_preopt: {
2094 enabled: false,
2095 },
2096 }
2097 `)
2098
2099 variant := ctx.ModuleForTests("foo", "android_common")
2100
2101 // Check dexpreopt outputs. They shouldn't exist.
2102 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2103 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2104 t.Errorf("dexpreopt shouldn't have run.")
2105 }
2106}
2107
2108func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002109 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002110 android_app_import {
2111 name: "foo",
2112 apk: "prebuilts/apk/app.apk",
2113 presigned: true,
2114 dex_preopt: {
2115 enabled: true,
2116 },
2117 }
2118 `)
2119
2120 variant := ctx.ModuleForTests("foo", "android_common")
2121
2122 // Check dexpreopt outputs.
2123 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2124 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2125 t.Errorf("can't find dexpreopt outputs")
2126 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002127 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002128 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2129 t.Errorf("signing rule shouldn't be included.")
2130 }
2131 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2132 t.Errorf("can't find aligning rule")
2133 }
2134}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002135
Liz Kammer2bc57f62020-05-13 15:49:21 -07002136func TestAndroidAppImport_SigningLineage(t *testing.T) {
2137 ctx, _ := testJava(t, `
2138 android_app_import {
2139 name: "foo",
2140 apk: "prebuilts/apk/app.apk",
2141 certificate: "platform",
2142 lineage: "lineage.bin",
2143 }
2144 `)
2145
2146 variant := ctx.ModuleForTests("foo", "android_common")
2147
2148 // Check cert signing lineage flag.
2149 signedApk := variant.Output("signed/foo.apk")
2150 signingFlag := signedApk.Args["flags"]
2151 expected := "--lineage lineage.bin"
2152 if expected != signingFlag {
2153 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2154 }
2155}
2156
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002157func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2158 ctx, _ := testJava(t, `
2159 android_app_import {
2160 name: "foo",
2161 apk: "prebuilts/apk/app.apk",
2162 default_dev_cert: true,
2163 dex_preopt: {
2164 enabled: true,
2165 },
2166 }
2167 `)
2168
2169 variant := ctx.ModuleForTests("foo", "android_common")
2170
2171 // Check dexpreopt outputs.
2172 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2173 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2174 t.Errorf("can't find dexpreopt outputs")
2175 }
2176
2177 // Check cert signing flag.
2178 signedApk := variant.Output("signed/foo.apk")
2179 signingFlag := signedApk.Args["certificates"]
2180 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2181 if expected != signingFlag {
2182 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2183 }
2184}
2185
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002186func TestAndroidAppImport_DpiVariants(t *testing.T) {
2187 bp := `
2188 android_app_import {
2189 name: "foo",
2190 apk: "prebuilts/apk/app.apk",
2191 dpi_variants: {
2192 xhdpi: {
2193 apk: "prebuilts/apk/app_xhdpi.apk",
2194 },
2195 xxhdpi: {
2196 apk: "prebuilts/apk/app_xxhdpi.apk",
2197 },
2198 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002199 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002200 dex_preopt: {
2201 enabled: true,
2202 },
2203 }
2204 `
2205 testCases := []struct {
2206 name string
2207 aaptPreferredConfig *string
2208 aaptPrebuiltDPI []string
2209 expected string
2210 }{
2211 {
2212 name: "no preferred",
2213 aaptPreferredConfig: nil,
2214 aaptPrebuiltDPI: []string{},
2215 expected: "prebuilts/apk/app.apk",
2216 },
2217 {
2218 name: "AAPTPreferredConfig matches",
2219 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002220 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002221 expected: "prebuilts/apk/app_xhdpi.apk",
2222 },
2223 {
2224 name: "AAPTPrebuiltDPI matches",
2225 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2226 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2227 expected: "prebuilts/apk/app_xxhdpi.apk",
2228 },
2229 {
2230 name: "non-first AAPTPrebuiltDPI matches",
2231 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2232 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2233 expected: "prebuilts/apk/app_xhdpi.apk",
2234 },
2235 {
2236 name: "no matches",
2237 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2238 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2239 expected: "prebuilts/apk/app.apk",
2240 },
2241 }
2242
2243 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2244 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002245 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002246 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2247 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002248 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002249
2250 run(t, ctx, config)
2251
2252 variant := ctx.ModuleForTests("foo", "android_common")
2253 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2254 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2255 if len(matches) != 2 {
2256 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2257 }
2258 if test.expected != matches[1] {
2259 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2260 }
2261 }
2262}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002263
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002264func TestAndroidAppImport_Filename(t *testing.T) {
2265 ctx, config := testJava(t, `
2266 android_app_import {
2267 name: "foo",
2268 apk: "prebuilts/apk/app.apk",
2269 presigned: true,
2270 }
2271
2272 android_app_import {
2273 name: "bar",
2274 apk: "prebuilts/apk/app.apk",
2275 presigned: true,
2276 filename: "bar_sample.apk"
2277 }
2278 `)
2279
2280 testCases := []struct {
2281 name string
2282 expected string
2283 }{
2284 {
2285 name: "foo",
2286 expected: "foo.apk",
2287 },
2288 {
2289 name: "bar",
2290 expected: "bar_sample.apk",
2291 },
2292 }
2293
2294 for _, test := range testCases {
2295 variant := ctx.ModuleForTests(test.name, "android_common")
2296 if variant.MaybeOutput(test.expected).Rule == nil {
2297 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2298 }
2299
2300 a := variant.Module().(*AndroidAppImport)
2301 expectedValues := []string{test.expected}
2302 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002303 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002304 if !reflect.DeepEqual(actualValues, expectedValues) {
2305 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2306 actualValues, expectedValues)
2307 }
2308 }
2309}
2310
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002311func TestAndroidAppImport_ArchVariants(t *testing.T) {
2312 // The test config's target arch is ARM64.
2313 testCases := []struct {
2314 name string
2315 bp string
2316 expected string
2317 }{
2318 {
2319 name: "matching arch",
2320 bp: `
2321 android_app_import {
2322 name: "foo",
2323 apk: "prebuilts/apk/app.apk",
2324 arch: {
2325 arm64: {
2326 apk: "prebuilts/apk/app_arm64.apk",
2327 },
2328 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002329 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002330 dex_preopt: {
2331 enabled: true,
2332 },
2333 }
2334 `,
2335 expected: "prebuilts/apk/app_arm64.apk",
2336 },
2337 {
2338 name: "no matching arch",
2339 bp: `
2340 android_app_import {
2341 name: "foo",
2342 apk: "prebuilts/apk/app.apk",
2343 arch: {
2344 arm: {
2345 apk: "prebuilts/apk/app_arm.apk",
2346 },
2347 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002348 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002349 dex_preopt: {
2350 enabled: true,
2351 },
2352 }
2353 `,
2354 expected: "prebuilts/apk/app.apk",
2355 },
2356 }
2357
2358 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2359 for _, test := range testCases {
2360 ctx, _ := testJava(t, test.bp)
2361
2362 variant := ctx.ModuleForTests("foo", "android_common")
2363 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2364 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2365 if len(matches) != 2 {
2366 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2367 }
2368 if test.expected != matches[1] {
2369 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2370 }
2371 }
2372}
2373
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002374func TestAndroidTestImport(t *testing.T) {
2375 ctx, config := testJava(t, `
2376 android_test_import {
2377 name: "foo",
2378 apk: "prebuilts/apk/app.apk",
2379 presigned: true,
2380 data: [
2381 "testdata/data",
2382 ],
2383 }
2384 `)
2385
2386 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2387
2388 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002389 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002390 expected := []string{"tests"}
2391 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2392 if !reflect.DeepEqual(expected, actual) {
2393 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2394 }
2395 expected = []string{"testdata/data:testdata/data"}
2396 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2397 if !reflect.DeepEqual(expected, actual) {
2398 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2399 }
2400}
2401
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002402func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2403 ctx, _ := testJava(t, `
2404 android_test_import {
2405 name: "foo",
2406 apk: "prebuilts/apk/app.apk",
2407 certificate: "cert/new_cert",
2408 data: [
2409 "testdata/data",
2410 ],
2411 }
2412
2413 android_test_import {
2414 name: "foo_presigned",
2415 apk: "prebuilts/apk/app.apk",
2416 presigned: true,
2417 data: [
2418 "testdata/data",
2419 ],
2420 }
2421 `)
2422
2423 variant := ctx.ModuleForTests("foo", "android_common")
2424 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2425 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2426 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2427 }
2428
2429 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2430 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2431 if jniRule != android.Cp.String() {
2432 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2433 }
Liz Kammer7e20dda2020-05-20 14:36:30 -07002434 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2435 t.Errorf("Presigned test apk should be aligned")
2436 }
2437}
2438
2439func TestAndroidTestImport_Preprocessed(t *testing.T) {
2440 ctx, _ := testJava(t, `
2441 android_test_import {
2442 name: "foo",
2443 apk: "prebuilts/apk/app.apk",
2444 presigned: true,
2445 preprocessed: true,
2446 }
2447
2448 android_test_import {
2449 name: "foo_cert",
2450 apk: "prebuilts/apk/app.apk",
2451 certificate: "cert/new_cert",
2452 preprocessed: true,
2453 }
2454 `)
2455
2456 testModules := []string{"foo", "foo_cert"}
2457 for _, m := range testModules {
2458 apkName := m + ".apk"
2459 variant := ctx.ModuleForTests(m, "android_common")
2460 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2461 if jniRule != android.Cp.String() {
2462 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2463 }
2464
2465 // Make sure signing and aligning were skipped.
2466 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2467 t.Errorf("signing rule shouldn't be included for preprocessed.")
2468 }
2469 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2470 t.Errorf("aligning rule shouldn't be for preprocessed")
2471 }
2472 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002473}
2474
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002475func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002476 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002477 cc_library {
2478 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002479 sdk_version: "current",
2480 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002481 }
2482
2483 android_test {
2484 name: "stl",
2485 jni_libs: ["libjni"],
2486 compile_multilib: "both",
2487 sdk_version: "current",
2488 stl: "c++_shared",
2489 }
2490
2491 android_test {
2492 name: "system",
2493 jni_libs: ["libjni"],
2494 compile_multilib: "both",
2495 sdk_version: "current",
2496 }
2497 `)
2498
2499 testCases := []struct {
2500 name string
2501 jnis []string
2502 }{
2503 {"stl",
2504 []string{
2505 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002506 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002507 },
2508 },
2509 {"system",
2510 []string{
2511 "libjni.so",
2512 },
2513 },
2514 }
2515
2516 for _, test := range testCases {
2517 t.Run(test.name, func(t *testing.T) {
2518 app := ctx.ModuleForTests(test.name, "android_common")
2519 jniLibZip := app.Output("jnilibs.zip")
2520 var jnis []string
2521 args := strings.Fields(jniLibZip.Args["jarArgs"])
2522 for i := 0; i < len(args); i++ {
2523 if args[i] == "-f" {
2524 jnis = append(jnis, args[i+1])
2525 i += 1
2526 }
2527 }
2528 jnisJoined := strings.Join(jnis, " ")
2529 for _, jni := range test.jnis {
2530 if !strings.Contains(jnisJoined, jni) {
2531 t.Errorf("missing jni %q in %q", jni, jnis)
2532 }
2533 }
2534 })
2535 }
2536}
Colin Cross50ddcc42019-05-16 12:28:22 -07002537
2538func TestUsesLibraries(t *testing.T) {
2539 bp := `
2540 java_sdk_library {
2541 name: "foo",
2542 srcs: ["a.java"],
2543 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002544 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002545 }
2546
2547 java_sdk_library {
Paul Duffin64e61992020-05-15 10:20:31 +01002548 name: "qux",
2549 srcs: ["a.java"],
2550 api_packages: ["qux"],
2551 sdk_version: "current",
2552 }
2553
2554 java_sdk_library {
2555 name: "quuz",
2556 srcs: ["a.java"],
2557 api_packages: ["quuz"],
2558 sdk_version: "current",
2559 }
2560
2561 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002562 name: "bar",
2563 srcs: ["a.java"],
2564 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002565 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002566 }
2567
2568 android_app {
2569 name: "app",
2570 srcs: ["a.java"],
Paul Duffin64e61992020-05-15 10:20:31 +01002571 libs: ["qux", "quuz.stubs"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002572 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002573 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002574 optional_uses_libs: [
2575 "bar",
2576 "baz",
2577 ],
2578 }
2579
2580 android_app_import {
2581 name: "prebuilt",
2582 apk: "prebuilts/apk/app.apk",
2583 certificate: "platform",
2584 uses_libs: ["foo"],
2585 optional_uses_libs: [
2586 "bar",
2587 "baz",
2588 ],
2589 }
2590 `
2591
Colin Cross98be1bb2019-12-13 20:41:13 -08002592 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002593 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2594
Colin Cross98be1bb2019-12-13 20:41:13 -08002595 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002596
2597 run(t, ctx, config)
2598
2599 app := ctx.ModuleForTests("app", "android_common")
2600 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2601
Paul Duffin64e61992020-05-15 10:20:31 +01002602 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2603 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2604 if w := "--uses-library qux"; !strings.Contains(manifestFixerArgs, w) {
2605 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2606 }
2607 if w := "--uses-library quuz"; !strings.Contains(manifestFixerArgs, w) {
2608 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2609 }
2610
Colin Cross50ddcc42019-05-16 12:28:22 -07002611 // Test that all libraries are verified
2612 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2613 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2614 t.Errorf("wanted %q in %q", w, cmd)
2615 }
2616
2617 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2618 t.Errorf("wanted %q in %q", w, cmd)
2619 }
2620
2621 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2622
2623 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2624 t.Errorf("wanted %q in %q", w, cmd)
2625 }
2626
2627 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2628 t.Errorf("wanted %q in %q", w, cmd)
2629 }
2630
2631 // Test that only present libraries are preopted
2632 cmd = app.Rule("dexpreopt").RuleParams.Command
2633
2634 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2635 t.Errorf("wanted %q in %q", w, cmd)
2636 }
2637
2638 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2639
2640 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2641 t.Errorf("wanted %q in %q", w, cmd)
2642 }
2643}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002644
2645func TestCodelessApp(t *testing.T) {
2646 testCases := []struct {
2647 name string
2648 bp string
2649 noCode bool
2650 }{
2651 {
2652 name: "normal",
2653 bp: `
2654 android_app {
2655 name: "foo",
2656 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002657 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002658 }
2659 `,
2660 noCode: false,
2661 },
2662 {
2663 name: "app without sources",
2664 bp: `
2665 android_app {
2666 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002667 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002668 }
2669 `,
2670 noCode: true,
2671 },
2672 {
2673 name: "app with libraries",
2674 bp: `
2675 android_app {
2676 name: "foo",
2677 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002678 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002679 }
2680
2681 java_library {
2682 name: "lib",
2683 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002684 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002685 }
2686 `,
2687 noCode: false,
2688 },
2689 {
2690 name: "app with sourceless libraries",
2691 bp: `
2692 android_app {
2693 name: "foo",
2694 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002695 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002696 }
2697
2698 java_library {
2699 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002700 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002701 }
2702 `,
2703 // TODO(jungjw): this should probably be true
2704 noCode: false,
2705 },
2706 }
2707
2708 for _, test := range testCases {
2709 t.Run(test.name, func(t *testing.T) {
2710 ctx := testApp(t, test.bp)
2711
2712 foo := ctx.ModuleForTests("foo", "android_common")
2713 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2714 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2715 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2716 }
2717 })
2718 }
2719}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002720
2721func TestEmbedNotice(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002722 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002723 android_app {
2724 name: "foo",
2725 srcs: ["a.java"],
2726 static_libs: ["javalib"],
2727 jni_libs: ["libjni"],
2728 notice: "APP_NOTICE",
2729 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002730 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002731 }
2732
2733 // No embed_notice flag
2734 android_app {
2735 name: "bar",
2736 srcs: ["a.java"],
2737 jni_libs: ["libjni"],
2738 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002739 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002740 }
2741
2742 // No NOTICE files
2743 android_app {
2744 name: "baz",
2745 srcs: ["a.java"],
2746 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002747 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002748 }
2749
2750 cc_library {
2751 name: "libjni",
2752 system_shared_libs: [],
2753 stl: "none",
2754 notice: "LIB_NOTICE",
Colin Cross1c93c292020-02-15 10:38:00 -08002755 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002756 }
2757
2758 java_library {
2759 name: "javalib",
2760 srcs: [
2761 ":gen",
2762 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002763 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002764 }
2765
2766 genrule {
2767 name: "gen",
2768 tools: ["gentool"],
2769 out: ["gen.java"],
2770 notice: "GENRULE_NOTICE",
2771 }
2772
2773 java_binary_host {
2774 name: "gentool",
2775 srcs: ["b.java"],
2776 notice: "TOOL_NOTICE",
2777 }
2778 `)
2779
2780 // foo has NOTICE files to process, and embed_notices is true.
2781 foo := ctx.ModuleForTests("foo", "android_common")
2782 // verify merge notices rule.
2783 mergeNotices := foo.Rule("mergeNoticesRule")
2784 noticeInputs := mergeNotices.Inputs.Strings()
2785 // TOOL_NOTICE should be excluded as it's a host module.
2786 if len(mergeNotices.Inputs) != 3 {
2787 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2788 }
2789 if !inList("APP_NOTICE", noticeInputs) {
2790 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2791 }
2792 if !inList("LIB_NOTICE", noticeInputs) {
2793 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2794 }
2795 if !inList("GENRULE_NOTICE", noticeInputs) {
2796 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2797 }
2798 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2799 res := foo.Output("package-res.apk")
2800 aapt2Flags := res.Args["flags"]
2801 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2802 if !strings.Contains(aapt2Flags, e) {
2803 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2804 }
2805
2806 // bar has NOTICE files to process, but embed_notices is not set.
2807 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002808 res = bar.Output("package-res.apk")
2809 aapt2Flags = res.Args["flags"]
2810 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2811 if strings.Contains(aapt2Flags, e) {
2812 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002813 }
2814
2815 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2816 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002817 res = baz.Output("package-res.apk")
2818 aapt2Flags = res.Args["flags"]
2819 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2820 if strings.Contains(aapt2Flags, e) {
2821 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002822 }
2823}
Colin Cross53a87f52019-06-25 13:35:30 -07002824
2825func TestUncompressDex(t *testing.T) {
2826 testCases := []struct {
2827 name string
2828 bp string
2829
2830 uncompressedPlatform bool
2831 uncompressedUnbundled bool
2832 }{
2833 {
2834 name: "normal",
2835 bp: `
2836 android_app {
2837 name: "foo",
2838 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002839 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002840 }
2841 `,
2842 uncompressedPlatform: true,
2843 uncompressedUnbundled: false,
2844 },
2845 {
2846 name: "use_embedded_dex",
2847 bp: `
2848 android_app {
2849 name: "foo",
2850 use_embedded_dex: true,
2851 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002852 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002853 }
2854 `,
2855 uncompressedPlatform: true,
2856 uncompressedUnbundled: true,
2857 },
2858 {
2859 name: "privileged",
2860 bp: `
2861 android_app {
2862 name: "foo",
2863 privileged: true,
2864 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002865 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002866 }
2867 `,
2868 uncompressedPlatform: true,
2869 uncompressedUnbundled: true,
2870 },
David Srbecky98c71222020-05-20 22:20:28 +01002871 {
2872 name: "normal_uncompress_dex_true",
2873 bp: `
2874 android_app {
2875 name: "foo",
2876 srcs: ["a.java"],
2877 sdk_version: "current",
2878 uncompress_dex: true,
2879 }
2880 `,
2881 uncompressedPlatform: true,
2882 uncompressedUnbundled: true,
2883 },
2884 {
2885 name: "normal_uncompress_dex_false",
2886 bp: `
2887 android_app {
2888 name: "foo",
2889 srcs: ["a.java"],
2890 sdk_version: "current",
2891 uncompress_dex: false,
2892 }
2893 `,
2894 uncompressedPlatform: false,
2895 uncompressedUnbundled: false,
2896 },
Colin Cross53a87f52019-06-25 13:35:30 -07002897 }
2898
2899 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2900 t.Helper()
2901
Colin Cross98be1bb2019-12-13 20:41:13 -08002902 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002903 if unbundled {
2904 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2905 }
2906
Colin Cross98be1bb2019-12-13 20:41:13 -08002907 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002908
2909 run(t, ctx, config)
2910
2911 foo := ctx.ModuleForTests("foo", "android_common")
2912 dex := foo.Rule("r8")
2913 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2914 aligned := foo.MaybeRule("zipalign").Rule != nil
2915
2916 if uncompressedInDexJar != want {
2917 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2918 }
2919
2920 if aligned != want {
2921 t.Errorf("want aligned %v, got %v", want, aligned)
2922 }
2923 }
2924
2925 for _, tt := range testCases {
2926 t.Run(tt.name, func(t *testing.T) {
2927 t.Run("platform", func(t *testing.T) {
2928 test(t, tt.bp, tt.uncompressedPlatform, false)
2929 })
2930 t.Run("unbundled", func(t *testing.T) {
2931 test(t, tt.bp, tt.uncompressedUnbundled, true)
2932 })
2933 })
2934 }
2935}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002936
2937func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2938 if expectedValue != "" {
2939 expectedFlag := "--" + flagName + " " + expectedValue
2940 if !strings.Contains(aapt2Flags, expectedFlag) {
2941 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2942 }
2943 } else {
2944 unexpectedFlag := "--" + flagName
2945 if strings.Contains(aapt2Flags, unexpectedFlag) {
2946 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2947 }
2948 }
2949}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002950
2951func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungca095d72020-04-09 16:15:30 -07002952 fs := map[string][]byte{
2953 "baz/res/res/values/strings.xml": nil,
2954 "bar/res/res/values/strings.xml": nil,
2955 }
2956 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002957 runtime_resource_overlay {
2958 name: "foo",
2959 certificate: "platform",
Liz Kammer7fe241f2020-05-19 16:15:25 -07002960 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002961 product_specific: true,
Jaewoong Jungca095d72020-04-09 16:15:30 -07002962 static_libs: ["bar"],
2963 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002964 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002965 }
2966
2967 runtime_resource_overlay {
2968 name: "foo_themed",
2969 certificate: "platform",
2970 product_specific: true,
2971 theme: "faza",
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002972 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002973 }
Jaewoong Jungca095d72020-04-09 16:15:30 -07002974
2975 android_library {
2976 name: "bar",
2977 resource_dirs: ["bar/res"],
2978 }
2979
2980 android_app {
2981 name: "baz",
2982 sdk_version: "current",
2983 resource_dirs: ["baz/res"],
2984 }
2985 `
2986 config := testAppConfig(nil, bp, fs)
2987 ctx := testContext()
2988 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002989
2990 m := ctx.ModuleForTests("foo", "android_common")
2991
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002992 // Check AAPT2 link flags.
2993 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2994 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2995 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2996 if len(absentFlags) > 0 {
2997 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2998 }
2999
Jaewoong Jungca095d72020-04-09 16:15:30 -07003000 // Check overlay.list output for static_libs dependency.
3001 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
3002 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
3003 if !inList(staticLibPackage, overlayList) {
3004 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
3005 }
3006
3007 // Check AAPT2 link flags for resource_libs dependency.
3008 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
3009 if !strings.Contains(aapt2Flags, resourceLibFlag) {
3010 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
3011 }
3012
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003013 // Check cert signing flag.
3014 signedApk := m.Output("signed/foo.apk")
Liz Kammer7fe241f2020-05-19 16:15:25 -07003015 lineageFlag := signedApk.Args["flags"]
3016 expectedLineageFlag := "--lineage lineage.bin"
3017 if expectedLineageFlag != lineageFlag {
3018 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
3019 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003020 signingFlag := signedApk.Args["certificates"]
3021 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
3022 if expected != signingFlag {
3023 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
3024 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003025 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3026 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003027 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
3028 if !reflect.DeepEqual(path, expectedPath) {
3029 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
3030 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003031
3032 // Check device location.
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003033 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003034 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003035 if !reflect.DeepEqual(path, expectedPath) {
3036 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3037 }
3038
3039 // A themed module has a different device location
3040 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003041 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3042 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003043 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3044 if !reflect.DeepEqual(path, expectedPath) {
3045 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3046 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003047
3048 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3049 expectedOverrides := []string{"foo"}
3050 if !reflect.DeepEqual(overrides, expectedOverrides) {
3051 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3052 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003053}
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07003054
3055func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3056 ctx, config := testJava(t, `
3057 java_defaults {
3058 name: "rro_defaults",
3059 theme: "default_theme",
3060 product_specific: true,
3061 aaptflags: ["--keep-raw-values"],
3062 }
3063
3064 runtime_resource_overlay {
3065 name: "foo_with_defaults",
3066 defaults: ["rro_defaults"],
3067 }
3068
3069 runtime_resource_overlay {
3070 name: "foo_barebones",
3071 }
3072 `)
3073
3074 //
3075 // RRO module with defaults
3076 //
3077 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3078
3079 // Check AAPT2 link flags.
3080 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3081 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3082 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3083 if len(absentFlags) > 0 {
3084 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3085 }
3086
3087 // Check device location.
3088 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3089 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3090 if !reflect.DeepEqual(path, expectedPath) {
3091 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3092 }
3093
3094 //
3095 // RRO module without defaults
3096 //
3097 m = ctx.ModuleForTests("foo_barebones", "android_common")
3098
3099 // Check AAPT2 link flags.
3100 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3101 unexpectedFlags := "--keep-raw-values"
3102 if inList(unexpectedFlags, aapt2Flags) {
3103 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3104 }
3105
3106 // Check device location.
3107 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3108 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3109 if !reflect.DeepEqual(path, expectedPath) {
3110 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3111 }
3112}
Roshan Piusb8307962020-04-27 09:42:27 -07003113
3114func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3115 ctx, _ := testJava(t, `
3116 runtime_resource_overlay {
3117 name: "foo_overlay",
3118 certificate: "platform",
3119 product_specific: true,
3120 sdk_version: "current",
3121 }
3122
3123 override_runtime_resource_overlay {
3124 name: "bar_overlay",
3125 base: "foo_overlay",
3126 package_name: "com.android.bar.overlay",
3127 target_package_name: "com.android.bar",
3128 }
3129 `)
3130
3131 expectedVariants := []struct {
3132 moduleName string
3133 variantName string
3134 apkPath string
3135 overrides []string
3136 targetVariant string
3137 packageFlag string
3138 targetPackageFlag string
3139 }{
3140 {
3141 variantName: "android_common",
3142 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3143 overrides: nil,
3144 targetVariant: "android_common",
3145 packageFlag: "",
3146 targetPackageFlag: "",
3147 },
3148 {
3149 variantName: "android_common_bar_overlay",
3150 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3151 overrides: []string{"foo_overlay"},
3152 targetVariant: "android_common_bar",
3153 packageFlag: "com.android.bar.overlay",
3154 targetPackageFlag: "com.android.bar",
3155 },
3156 }
3157 for _, expected := range expectedVariants {
3158 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3159
3160 // Check the final apk name
3161 outputs := variant.AllOutputs()
3162 expectedApkPath := buildDir + expected.apkPath
3163 found := false
3164 for _, o := range outputs {
3165 if o == expectedApkPath {
3166 found = true
3167 break
3168 }
3169 }
3170 if !found {
3171 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3172 }
3173
3174 // Check if the overrides field values are correctly aggregated.
3175 mod := variant.Module().(*RuntimeResourceOverlay)
3176 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3177 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3178 expected.overrides, mod.properties.Overrides)
3179 }
3180
3181 // Check aapt2 flags.
3182 res := variant.Output("package-res.apk")
3183 aapt2Flags := res.Args["flags"]
3184 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
3185 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3186 }
3187}