blob: e0c5820ea1b8d31009692e26b27160f3d6fc873c [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 Smundaka7856c02020-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 Smundak18d98bc2020-05-27 16:36:07 -0700152 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-04-23 09:49:59 -0700153 params := module.Output(packedSplitApks)
154 if params.Rule == nil {
155 t.Errorf("expected output %s is missing", packedSplitApks)
156 }
157 if s := params.Args["allow-prereleased"]; s != "true" {
158 t.Errorf("wrong allow-prereleased value: '%s', expected 'true'", s)
159 }
160 mkEntries := android.AndroidMkEntriesForTest(t, config, "", module.Module())[0]
161 actualMaster := mkEntries.EntryMap["LOCAL_APK_SET_MASTER_FILE"]
162 expectedMaster := []string{"foo.apk"}
163 if !reflect.DeepEqual(actualMaster, expectedMaster) {
164 t.Errorf("Unexpected LOCAL_APK_SET_MASTER_FILE value: '%s', expected: '%s',",
165 actualMaster, expectedMaster)
166 }
167}
168
169func TestAndroidAppSet_Variants(t *testing.T) {
170 bp := `
171 android_app_set {
172 name: "foo",
173 set: "prebuilts/apks/app.apks",
174 }`
175 testCases := []struct {
176 name string
177 deviceArch *string
178 deviceSecondaryArch *string
179 aaptPrebuiltDPI []string
180 sdkVersion int
181 expected map[string]string
182 }{
183 {
184 name: "One",
185 deviceArch: proptools.StringPtr("x86"),
186 aaptPrebuiltDPI: []string{"ldpi", "xxhdpi"},
187 sdkVersion: 29,
188 expected: map[string]string{
189 "abis": "X86",
190 "allow-prereleased": "false",
191 "screen-densities": "LDPI,XXHDPI",
192 "sdk-version": "29",
193 "stem": "foo",
194 },
195 },
196 {
197 name: "Two",
198 deviceArch: proptools.StringPtr("x86_64"),
199 deviceSecondaryArch: proptools.StringPtr("x86"),
200 aaptPrebuiltDPI: nil,
201 sdkVersion: 30,
202 expected: map[string]string{
203 "abis": "X86_64,X86",
204 "allow-prereleased": "false",
205 "screen-densities": "all",
206 "sdk-version": "30",
207 "stem": "foo",
208 },
209 },
210 }
211
212 for _, test := range testCases {
213 config := testAppConfig(nil, bp, nil)
214 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
215 config.TestProductVariables.Platform_sdk_version = &test.sdkVersion
216 config.TestProductVariables.DeviceArch = test.deviceArch
217 config.TestProductVariables.DeviceSecondaryArch = test.deviceSecondaryArch
218 ctx := testContext()
219 run(t, ctx, config)
220 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundak18d98bc2020-05-27 16:36:07 -0700221 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-04-23 09:49:59 -0700222 params := module.Output(packedSplitApks)
223 for k, v := range test.expected {
224 if actual := params.Args[k]; actual != v {
225 t.Errorf("%s: bad build arg value for '%s': '%s', expected '%s'",
226 test.name, k, actual, v)
227 }
228 }
229 }
230}
231
Jeongik Cha538c0d02019-07-11 15:54:27 +0900232func TestPlatformAPIs(t *testing.T) {
233 testJava(t, `
234 android_app {
235 name: "foo",
236 srcs: ["a.java"],
237 platform_apis: true,
238 }
239 `)
240
241 testJava(t, `
242 android_app {
243 name: "foo",
244 srcs: ["a.java"],
245 sdk_version: "current",
246 }
247 `)
248
249 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
250 android_app {
251 name: "bar",
252 srcs: ["b.java"],
253 }
254 `)
255
256 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
257 android_app {
258 name: "bar",
259 srcs: ["b.java"],
260 sdk_version: "system_current",
261 platform_apis: true,
262 }
263 `)
264}
265
Jeongik Chae403e9e2019-12-07 00:16:24 +0900266func TestAndroidAppLinkType(t *testing.T) {
267 testJava(t, `
268 android_app {
269 name: "foo",
270 srcs: ["a.java"],
271 libs: ["bar"],
272 static_libs: ["baz"],
273 platform_apis: true,
274 }
275
276 java_library {
277 name: "bar",
278 sdk_version: "current",
279 srcs: ["b.java"],
280 }
281
282 android_library {
283 name: "baz",
284 sdk_version: "system_current",
285 srcs: ["c.java"],
286 }
287 `)
288
289 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
290 android_app {
291 name: "foo",
292 srcs: ["a.java"],
293 libs: ["bar"],
294 sdk_version: "current",
295 static_libs: ["baz"],
296 }
297
298 java_library {
299 name: "bar",
300 sdk_version: "current",
301 srcs: ["b.java"],
302 }
303
304 android_library {
305 name: "baz",
306 sdk_version: "system_current",
307 srcs: ["c.java"],
308 }
309 `)
310
311 testJava(t, `
312 android_app {
313 name: "foo",
314 srcs: ["a.java"],
315 libs: ["bar"],
316 sdk_version: "system_current",
317 static_libs: ["baz"],
318 }
319
320 java_library {
321 name: "bar",
322 sdk_version: "current",
323 srcs: ["b.java"],
324 }
325
326 android_library {
327 name: "baz",
328 sdk_version: "system_current",
329 srcs: ["c.java"],
330 }
331 `)
332
333 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
334 android_app {
335 name: "foo",
336 srcs: ["a.java"],
337 libs: ["bar"],
338 sdk_version: "system_current",
339 static_libs: ["baz"],
340 }
341
342 java_library {
343 name: "bar",
344 sdk_version: "current",
345 srcs: ["b.java"],
346 }
347
348 android_library {
349 name: "baz",
350 srcs: ["c.java"],
351 }
352 `)
353}
354
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100355func TestUpdatableApps(t *testing.T) {
356 testCases := []struct {
357 name string
358 bp string
359 expectedError string
360 }{
361 {
362 name: "Stable public SDK",
363 bp: `android_app {
364 name: "foo",
365 srcs: ["a.java"],
366 sdk_version: "29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100367 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100368 updatable: true,
369 }`,
370 },
371 {
372 name: "Stable system SDK",
373 bp: `android_app {
374 name: "foo",
375 srcs: ["a.java"],
376 sdk_version: "system_29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100377 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100378 updatable: true,
379 }`,
380 },
381 {
382 name: "Current public SDK",
383 bp: `android_app {
384 name: "foo",
385 srcs: ["a.java"],
386 sdk_version: "current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100387 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100388 updatable: true,
389 }`,
390 },
391 {
392 name: "Current system SDK",
393 bp: `android_app {
394 name: "foo",
395 srcs: ["a.java"],
396 sdk_version: "system_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100397 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100398 updatable: true,
399 }`,
400 },
401 {
402 name: "Current module SDK",
403 bp: `android_app {
404 name: "foo",
405 srcs: ["a.java"],
406 sdk_version: "module_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100407 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100408 updatable: true,
409 }`,
410 },
411 {
412 name: "Current core SDK",
413 bp: `android_app {
414 name: "foo",
415 srcs: ["a.java"],
416 sdk_version: "core_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100417 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100418 updatable: true,
419 }`,
420 },
421 {
422 name: "No Platform APIs",
423 bp: `android_app {
424 name: "foo",
425 srcs: ["a.java"],
426 platform_apis: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100427 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100428 updatable: true,
429 }`,
430 expectedError: "Updatable apps must use stable SDKs",
431 },
432 {
433 name: "No Core Platform APIs",
434 bp: `android_app {
435 name: "foo",
436 srcs: ["a.java"],
437 sdk_version: "core_platform",
Artur Satayevf40fc852020-04-16 13:43:02 +0100438 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100439 updatable: true,
440 }`,
441 expectedError: "Updatable apps must use stable SDKs",
442 },
443 {
444 name: "No unspecified APIs",
445 bp: `android_app {
446 name: "foo",
447 srcs: ["a.java"],
448 updatable: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100449 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100450 }`,
451 expectedError: "Updatable apps must use stable SDK",
452 },
Artur Satayevf40fc852020-04-16 13:43:02 +0100453 {
454 name: "Must specify min_sdk_version",
455 bp: `android_app {
456 name: "app_without_min_sdk_version",
457 srcs: ["a.java"],
458 sdk_version: "29",
459 updatable: true,
460 }`,
461 expectedError: "updatable apps must set min_sdk_version.",
462 },
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100463 }
464
465 for _, test := range testCases {
466 t.Run(test.name, func(t *testing.T) {
467 if test.expectedError == "" {
468 testJava(t, test.bp)
469 } else {
470 testJavaError(t, test.expectedError, test.bp)
471 }
472 })
473 }
474}
475
Jooyung Han749dc692020-04-15 11:03:39 +0900476func TestUpdatableApps_TransitiveDepsShouldSetMinSdkVersion(t *testing.T) {
477 testJavaError(t, `module "bar".*: should support min_sdk_version\(29\)`, cc.GatherRequiredDepsForTest(android.Android)+`
478 android_app {
479 name: "foo",
480 srcs: ["a.java"],
481 updatable: true,
482 sdk_version: "current",
483 min_sdk_version: "29",
484 static_libs: ["bar"],
485 }
486
487 java_library {
488 name: "bar",
489 sdk_version: "current",
490 }
491 `)
492}
493
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900494func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
495 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
496 android_app {
497 name: "foo",
498 srcs: ["a.java"],
499 updatable: true,
500 sdk_version: "current",
501 min_sdk_version: "current",
502 jni_libs: ["libjni"],
503 }
504
505 cc_library {
506 name: "libjni",
507 stl: "none",
508 system_shared_libs: [],
509 sdk_version: "current",
510 }
511 `)
512}
513
514func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
515 bp := cc.GatherRequiredDepsForTest(android.Android) + `
516 android_app {
517 name: "foo",
518 srcs: ["a.java"],
519 updatable: true,
520 sdk_version: "current",
521 min_sdk_version: "29",
522 jni_libs: ["libjni"],
523 }
524
525 cc_library {
526 name: "libjni",
527 stl: "none",
528 system_shared_libs: [],
529 sdk_version: "29",
530 }
531
532 ndk_prebuilt_object {
533 name: "ndk_crtbegin_so.29",
534 sdk_version: "29",
535 }
536
537 ndk_prebuilt_object {
538 name: "ndk_crtend_so.29",
539 sdk_version: "29",
540 }
541 `
542 fs := map[string][]byte{
543 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
544 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
545 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
546 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
547 }
548
549 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
550
551 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
552 var crtbeginFound, crtendFound bool
553 for _, input := range inputs {
554 switch input.String() {
555 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o":
556 crtbeginFound = true
557 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o":
558 crtendFound = true
559 }
560 }
561 if !crtbeginFound || !crtendFound {
562 t.Error("should link with ndk_crtbegin_so.29 and ndk_crtend_so.29")
563 }
564}
565
566func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
567 bp := cc.GatherRequiredDepsForTest(android.Android) + `
568 android_app {
569 name: "foo",
570 srcs: ["a.java"],
571 updatable: true,
572 sdk_version: "current",
573 min_sdk_version: "29", // this APK should support 29
574 jni_libs: ["libjni"],
575 }
576
577 cc_library {
578 name: "libjni",
579 stl: "none",
580 sdk_version: "current",
581 }
582 `
583 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
584}
585
586func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
587 bp := cc.GatherRequiredDepsForTest(android.Android) + `
588 android_app {
589 name: "foo",
590 srcs: ["a.java"],
591 updatable: true,
592 sdk_version: "current",
593 min_sdk_version: "29", // this APK should support 29
594 jni_libs: ["libjni"],
595 }
596
597 cc_library {
598 name: "libjni",
599 stl: "none",
600 shared_libs: ["libbar"],
601 system_shared_libs: [],
602 sdk_version: "27",
603 }
604
605 cc_library {
606 name: "libbar",
607 stl: "none",
608 system_shared_libs: [],
609 sdk_version: "current",
610 }
611 `
612 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
613}
614
Colin Cross0ddae7f2019-02-07 15:30:01 -0800615func TestResourceDirs(t *testing.T) {
616 testCases := []struct {
617 name string
618 prop string
619 resources []string
620 }{
621 {
622 name: "no resource_dirs",
623 prop: "",
624 resources: []string{"res/res/values/strings.xml"},
625 },
626 {
627 name: "resource_dirs",
628 prop: `resource_dirs: ["res"]`,
629 resources: []string{"res/res/values/strings.xml"},
630 },
631 {
632 name: "empty resource_dirs",
633 prop: `resource_dirs: []`,
634 resources: nil,
635 },
636 }
637
638 fs := map[string][]byte{
639 "res/res/values/strings.xml": nil,
640 }
641
642 bp := `
643 android_app {
644 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900645 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800646 %s
647 }
648 `
649
650 for _, testCase := range testCases {
651 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800652 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
653 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800654 run(t, ctx, config)
655
656 module := ctx.ModuleForTests("foo", "android_common")
657 resourceList := module.MaybeOutput("aapt2/res.list")
658
659 var resources []string
660 if resourceList.Rule != nil {
661 for _, compiledResource := range resourceList.Inputs.Strings() {
662 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
663 }
664 }
665
666 if !reflect.DeepEqual(resources, testCase.resources) {
667 t.Errorf("expected resource files %q, got %q",
668 testCase.resources, resources)
669 }
670 })
671 }
672}
673
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800674func TestLibraryAssets(t *testing.T) {
675 bp := `
676 android_app {
677 name: "foo",
678 sdk_version: "current",
679 static_libs: ["lib1", "lib2", "lib3"],
680 }
681
682 android_library {
683 name: "lib1",
684 sdk_version: "current",
685 asset_dirs: ["assets_a"],
686 }
687
688 android_library {
689 name: "lib2",
690 sdk_version: "current",
691 }
692
693 android_library {
694 name: "lib3",
695 sdk_version: "current",
696 static_libs: ["lib4"],
697 }
698
699 android_library {
700 name: "lib4",
701 sdk_version: "current",
702 asset_dirs: ["assets_b"],
703 }
704 `
705
706 testCases := []struct {
707 name string
708 assetFlag string
709 assetPackages []string
710 }{
711 {
712 name: "foo",
713 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
714 assetPackages: []string{
715 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
716 buildDir + "/.intermediates/lib1/android_common/assets.zip",
717 buildDir + "/.intermediates/lib3/android_common/assets.zip",
718 },
719 },
720 {
721 name: "lib1",
722 assetFlag: "-A assets_a",
723 },
724 {
725 name: "lib2",
726 },
727 {
728 name: "lib3",
729 assetPackages: []string{
730 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
731 buildDir + "/.intermediates/lib4/android_common/assets.zip",
732 },
733 },
734 {
735 name: "lib4",
736 assetFlag: "-A assets_b",
737 },
738 }
739 ctx := testApp(t, bp)
740
741 for _, test := range testCases {
742 t.Run(test.name, func(t *testing.T) {
743 m := ctx.ModuleForTests(test.name, "android_common")
744
745 // Check asset flag in aapt2 link flags
746 var aapt2link android.TestingBuildParams
747 if len(test.assetPackages) > 0 {
748 aapt2link = m.Output("aapt2/package-res.apk")
749 } else {
750 aapt2link = m.Output("package-res.apk")
751 }
752 aapt2Flags := aapt2link.Args["flags"]
753 if test.assetFlag != "" {
754 if !strings.Contains(aapt2Flags, test.assetFlag) {
755 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
756 }
757 } else {
758 if strings.Contains(aapt2Flags, " -A ") {
759 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
760 }
761 }
762
763 // Check asset merge rule.
764 if len(test.assetPackages) > 0 {
765 mergeAssets := m.Output("package-res.apk")
766 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
767 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
768 mergeAssets.Inputs.Strings(), test.assetPackages)
769 }
770 }
771 })
772 }
773}
774
Colin Crossbec85302019-02-13 13:15:46 -0800775func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800776 testCases := []struct {
777 name string
778 enforceRROTargets []string
779 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800780 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800781 overlayFiles map[string][]string
782 rroDirs map[string][]string
783 }{
784 {
785 name: "no RRO",
786 enforceRROTargets: nil,
787 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800788 resourceFiles: map[string][]string{
789 "foo": nil,
790 "bar": {"bar/res/res/values/strings.xml"},
791 "lib": nil,
792 "lib2": {"lib2/res/res/values/strings.xml"},
793 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800794 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800795 "foo": {
796 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800797 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000798 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800799 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800800 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
801 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000802 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800803 },
Colin Crossbec85302019-02-13 13:15:46 -0800804 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800805 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
806 "device/vendor/blah/overlay/bar/res/values/strings.xml",
807 },
Colin Crossbec85302019-02-13 13:15:46 -0800808 "lib": {
809 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
810 "lib/res/res/values/strings.xml",
811 "device/vendor/blah/overlay/lib/res/values/strings.xml",
812 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800813 },
814 rroDirs: map[string][]string{
815 "foo": nil,
816 "bar": nil,
817 },
818 },
819 {
820 name: "enforce RRO on foo",
821 enforceRROTargets: []string{"foo"},
822 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800823 resourceFiles: map[string][]string{
824 "foo": nil,
825 "bar": {"bar/res/res/values/strings.xml"},
826 "lib": nil,
827 "lib2": {"lib2/res/res/values/strings.xml"},
828 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800829 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800830 "foo": {
831 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800832 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000833 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800834 "foo/res/res/values/strings.xml",
835 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
836 },
Colin Crossbec85302019-02-13 13:15:46 -0800837 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800838 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
839 "device/vendor/blah/overlay/bar/res/values/strings.xml",
840 },
Colin Crossbec85302019-02-13 13:15:46 -0800841 "lib": {
842 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
843 "lib/res/res/values/strings.xml",
844 "device/vendor/blah/overlay/lib/res/values/strings.xml",
845 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800846 },
Colin Crossc1c37552019-01-31 11:42:41 -0800847
Colin Cross5c4791c2019-02-01 11:44:44 -0800848 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800849 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000850 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800851 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
852 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000853 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800854 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800855 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800856 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800857 },
858 },
859 {
860 name: "enforce RRO on all",
861 enforceRROTargets: []string{"*"},
862 enforceRROExcludedOverlays: []string{
863 // Excluding specific apps/res directories also allowed.
864 "device/vendor/blah/static_overlay/foo",
865 "device/vendor/blah/static_overlay/bar/res",
866 },
Colin Crossbec85302019-02-13 13:15:46 -0800867 resourceFiles: map[string][]string{
868 "foo": nil,
869 "bar": {"bar/res/res/values/strings.xml"},
870 "lib": nil,
871 "lib2": {"lib2/res/res/values/strings.xml"},
872 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800873 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800874 "foo": {
875 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800876 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000877 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800878 "foo/res/res/values/strings.xml",
879 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
880 },
Colin Crossbec85302019-02-13 13:15:46 -0800881 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
882 "lib": {
883 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
884 "lib/res/res/values/strings.xml",
885 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800886 },
887 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800888 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000889 "device:device/vendor/blah/overlay/foo/res",
890 "product:product/vendor/blah/overlay/foo/res",
891 // Lib dep comes after the direct deps
892 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800893 },
Anton Hansson53c88442019-03-18 15:53:16 +0000894 "bar": {"device:device/vendor/blah/overlay/bar/res"},
895 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800896 },
897 },
898 }
899
Anton Hansson53c88442019-03-18 15:53:16 +0000900 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800901 "device/vendor/blah/overlay",
902 "device/vendor/blah/overlay2",
903 "device/vendor/blah/static_overlay",
904 }
905
Anton Hansson53c88442019-03-18 15:53:16 +0000906 productResourceOverlays := []string{
907 "product/vendor/blah/overlay",
908 }
909
Colin Cross890ff552017-11-30 20:13:19 -0800910 fs := map[string][]byte{
911 "foo/res/res/values/strings.xml": nil,
912 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800913 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800914 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800915 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
916 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800917 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800918 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
919 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
920 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000921 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800922 }
923
924 bp := `
925 android_app {
926 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900927 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800928 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000929 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800930 }
931
932 android_app {
933 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900934 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800935 resource_dirs: ["bar/res"],
936 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800937
938 android_library {
939 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900940 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800941 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800942 static_libs: ["lib2"],
943 }
944
945 android_library {
946 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900947 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800948 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800949 }
Anton Hansson53c88442019-03-18 15:53:16 +0000950
951 // This library has the same resources as lib (should not lead to dupe RROs)
952 android_library {
953 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900954 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000955 resource_dirs: ["lib/res"]
956 }
Colin Cross890ff552017-11-30 20:13:19 -0800957 `
958
Colin Cross5c4791c2019-02-01 11:44:44 -0800959 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800960 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800961 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000962 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
963 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800964 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800965 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800966 }
967 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800968 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800969 }
970
Colin Cross98be1bb2019-12-13 20:41:13 -0800971 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800972 run(t, ctx, config)
973
Colin Crossbec85302019-02-13 13:15:46 -0800974 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
975 for _, o := range list {
976 res := module.MaybeOutput(o)
977 if res.Rule != nil {
978 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
979 // verify the inputs to the .arsc.flat rule.
980 files = append(files, res.Inputs.Strings()...)
981 } else {
982 // Otherwise, verify the full path to the output of the other module
983 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000984 }
Colin Cross890ff552017-11-30 20:13:19 -0800985 }
Colin Crossbec85302019-02-13 13:15:46 -0800986 return files
Colin Cross890ff552017-11-30 20:13:19 -0800987 }
988
Colin Crossbec85302019-02-13 13:15:46 -0800989 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
990 module := ctx.ModuleForTests(moduleName, "android_common")
991 resourceList := module.MaybeOutput("aapt2/res.list")
992 if resourceList.Rule != nil {
993 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000994 }
Colin Crossbec85302019-02-13 13:15:46 -0800995 overlayList := module.MaybeOutput("aapt2/overlay.list")
996 if overlayList.Rule != nil {
997 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
998 }
999
Anton Hansson53c88442019-03-18 15:53:16 +00001000 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
1001 var prefix string
1002 if d.overlayType == device {
1003 prefix = "device:"
1004 } else if d.overlayType == product {
1005 prefix = "product:"
1006 } else {
1007 t.Fatalf("Unexpected overlayType %d", d.overlayType)
1008 }
1009 rroDirs = append(rroDirs, prefix+d.path.String())
1010 }
Colin Crossbec85302019-02-13 13:15:46 -08001011
1012 return resourceFiles, overlayFiles, rroDirs
1013 }
1014
1015 modules := []string{"foo", "bar", "lib", "lib2"}
1016 for _, module := range modules {
1017 resourceFiles, overlayFiles, rroDirs := getResources(module)
1018
1019 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
1020 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
1021 module, testCase.resourceFiles[module], resourceFiles)
1022 }
1023 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
1024 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
1025 module, testCase.overlayFiles[module], overlayFiles)
1026 }
1027 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +00001028 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -08001029 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +00001030 }
Colin Cross890ff552017-11-30 20:13:19 -08001031 }
Colin Cross890ff552017-11-30 20:13:19 -08001032 })
1033 }
1034}
Colin Crossd09b0b62018-04-18 11:06:47 -07001035
1036func TestAppSdkVersion(t *testing.T) {
1037 testCases := []struct {
1038 name string
1039 sdkVersion string
1040 platformSdkInt int
1041 platformSdkCodename string
1042 platformSdkFinal bool
1043 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +09001044 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -07001045 }{
1046 {
1047 name: "current final SDK",
1048 sdkVersion: "current",
1049 platformSdkInt: 27,
1050 platformSdkCodename: "REL",
1051 platformSdkFinal: true,
1052 expectedMinSdkVersion: "27",
1053 },
1054 {
1055 name: "current non-final SDK",
1056 sdkVersion: "current",
1057 platformSdkInt: 27,
1058 platformSdkCodename: "OMR1",
1059 platformSdkFinal: false,
1060 expectedMinSdkVersion: "OMR1",
1061 },
1062 {
1063 name: "default final SDK",
1064 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001065 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001066 platformSdkInt: 27,
1067 platformSdkCodename: "REL",
1068 platformSdkFinal: true,
1069 expectedMinSdkVersion: "27",
1070 },
1071 {
1072 name: "default non-final SDK",
1073 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001074 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001075 platformSdkInt: 27,
1076 platformSdkCodename: "OMR1",
1077 platformSdkFinal: false,
1078 expectedMinSdkVersion: "OMR1",
1079 },
1080 {
1081 name: "14",
1082 sdkVersion: "14",
1083 expectedMinSdkVersion: "14",
1084 },
1085 }
1086
1087 for _, moduleType := range []string{"android_app", "android_library"} {
1088 for _, test := range testCases {
1089 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001090 platformApiProp := ""
1091 if test.platformApis {
1092 platformApiProp = "platform_apis: true,"
1093 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001094 bp := fmt.Sprintf(`%s {
1095 name: "foo",
1096 srcs: ["a.java"],
1097 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001098 %s
1099 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001100
Colin Cross98be1bb2019-12-13 20:41:13 -08001101 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001102 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1103 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1104 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1105
Colin Cross98be1bb2019-12-13 20:41:13 -08001106 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -07001107
1108 run(t, ctx, config)
1109
1110 foo := ctx.ModuleForTests("foo", "android_common")
1111 link := foo.Output("package-res.apk")
1112 linkFlags := strings.Split(link.Args["flags"], " ")
1113 min := android.IndexList("--min-sdk-version", linkFlags)
1114 target := android.IndexList("--target-sdk-version", linkFlags)
1115
1116 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1117 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1118 }
1119
1120 gotMinSdkVersion := linkFlags[min+1]
1121 gotTargetSdkVersion := linkFlags[target+1]
1122
1123 if gotMinSdkVersion != test.expectedMinSdkVersion {
1124 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1125 test.expectedMinSdkVersion, gotMinSdkVersion)
1126 }
1127
1128 if gotTargetSdkVersion != test.expectedMinSdkVersion {
1129 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1130 test.expectedMinSdkVersion, gotTargetSdkVersion)
1131 }
1132 })
1133 }
1134 }
1135}
Colin Crossa4f08812018-10-02 22:03:40 -07001136
Paul Duffin50c217c2019-06-12 13:25:22 +01001137func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001138 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001139 cc_library {
1140 name: "libjni",
1141 system_shared_libs: [],
Colin Crossc511bc52020-04-07 16:50:32 +00001142 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001143 stl: "none",
1144 }
1145
1146 android_test {
1147 name: "test",
1148 sdk_version: "core_platform",
1149 jni_libs: ["libjni"],
1150 }
1151
1152 android_test {
1153 name: "test_first",
1154 sdk_version: "core_platform",
1155 compile_multilib: "first",
1156 jni_libs: ["libjni"],
1157 }
1158
1159 android_test {
1160 name: "test_both",
1161 sdk_version: "core_platform",
1162 compile_multilib: "both",
1163 jni_libs: ["libjni"],
1164 }
1165
1166 android_test {
1167 name: "test_32",
1168 sdk_version: "core_platform",
1169 compile_multilib: "32",
1170 jni_libs: ["libjni"],
1171 }
1172
1173 android_test {
1174 name: "test_64",
1175 sdk_version: "core_platform",
1176 compile_multilib: "64",
1177 jni_libs: ["libjni"],
1178 }
1179 `)
1180
1181 testCases := []struct {
1182 name string
1183 abis []string
1184 }{
1185 {"test", []string{"arm64-v8a"}},
1186 {"test_first", []string{"arm64-v8a"}},
1187 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1188 {"test_32", []string{"armeabi-v7a"}},
1189 {"test_64", []string{"arm64-v8a"}},
1190 }
1191
1192 for _, test := range testCases {
1193 t.Run(test.name, func(t *testing.T) {
1194 app := ctx.ModuleForTests(test.name, "android_common")
1195 jniLibZip := app.Output("jnilibs.zip")
1196 var abis []string
1197 args := strings.Fields(jniLibZip.Args["jarArgs"])
1198 for i := 0; i < len(args); i++ {
1199 if args[i] == "-P" {
1200 abis = append(abis, filepath.Base(args[i+1]))
1201 i++
1202 }
1203 }
1204 if !reflect.DeepEqual(abis, test.abis) {
1205 t.Errorf("want abis %v, got %v", test.abis, abis)
1206 }
1207 })
1208 }
1209}
1210
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001211func TestAppSdkVersionByPartition(t *testing.T) {
1212 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1213 android_app {
1214 name: "foo",
1215 srcs: ["a.java"],
1216 vendor: true,
1217 platform_apis: true,
1218 }
1219 `)
1220
1221 testJava(t, `
1222 android_app {
1223 name: "bar",
1224 srcs: ["b.java"],
1225 platform_apis: true,
1226 }
1227 `)
1228
1229 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001230 bp := `
1231 android_app {
1232 name: "foo",
1233 srcs: ["a.java"],
1234 product_specific: true,
1235 platform_apis: true,
1236 }
1237 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001238
1239 config := testAppConfig(nil, bp, nil)
1240 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001241 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001242 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 +09001243 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001244 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001245 }
1246 }
1247}
1248
Paul Duffin50c217c2019-06-12 13:25:22 +01001249func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001250 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001251 cc_library {
1252 name: "libjni",
1253 system_shared_libs: [],
1254 stl: "none",
Colin Cross094cde42020-02-15 10:38:00 -08001255 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001256 }
1257
1258 android_app {
1259 name: "app",
1260 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001261 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001262 }
1263
1264 android_app {
1265 name: "app_noembed",
1266 jni_libs: ["libjni"],
1267 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001268 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001269 }
1270
1271 android_app {
1272 name: "app_embed",
1273 jni_libs: ["libjni"],
1274 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001275 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001276 }
1277
1278 android_test {
1279 name: "test",
Colin Crossc511bc52020-04-07 16:50:32 +00001280 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001281 jni_libs: ["libjni"],
1282 }
1283
1284 android_test {
1285 name: "test_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001286 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001287 jni_libs: ["libjni"],
1288 use_embedded_native_libs: false,
1289 }
1290
1291 android_test_helper_app {
1292 name: "test_helper",
Colin Crossc511bc52020-04-07 16:50:32 +00001293 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001294 jni_libs: ["libjni"],
1295 }
1296
1297 android_test_helper_app {
1298 name: "test_helper_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001299 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001300 jni_libs: ["libjni"],
1301 use_embedded_native_libs: false,
1302 }
1303 `)
1304
1305 testCases := []struct {
1306 name string
1307 packaged bool
1308 compressed bool
1309 }{
1310 {"app", false, false},
1311 {"app_noembed", false, false},
1312 {"app_embed", true, false},
1313 {"test", true, false},
1314 {"test_noembed", true, true},
1315 {"test_helper", true, false},
1316 {"test_helper_noembed", true, true},
1317 }
1318
1319 for _, test := range testCases {
1320 t.Run(test.name, func(t *testing.T) {
1321 app := ctx.ModuleForTests(test.name, "android_common")
1322 jniLibZip := app.MaybeOutput("jnilibs.zip")
1323 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1324 t.Errorf("expected jni packaged %v, got %v", w, g)
1325 }
1326
1327 if jniLibZip.Rule != nil {
1328 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1329 t.Errorf("expected jni compressed %v, got %v", w, g)
1330 }
Colin Crossc511bc52020-04-07 16:50:32 +00001331
1332 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1333 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1334 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001335 }
1336 })
1337 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001338}
1339
Colin Cross3c007702020-05-08 11:20:24 -07001340func TestJNISDK(t *testing.T) {
1341 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1342 cc_library {
1343 name: "libjni",
1344 system_shared_libs: [],
1345 stl: "none",
1346 sdk_version: "current",
1347 }
1348
1349 android_test {
1350 name: "app_platform",
1351 jni_libs: ["libjni"],
1352 platform_apis: true,
1353 }
1354
1355 android_test {
1356 name: "app_sdk",
1357 jni_libs: ["libjni"],
1358 sdk_version: "current",
1359 }
1360
1361 android_test {
1362 name: "app_force_platform",
1363 jni_libs: ["libjni"],
1364 sdk_version: "current",
1365 jni_uses_platform_apis: true,
1366 }
1367
1368 android_test {
1369 name: "app_force_sdk",
1370 jni_libs: ["libjni"],
1371 platform_apis: true,
1372 jni_uses_sdk_apis: true,
1373 }
Colin Crossc2d24052020-05-13 11:05:02 -07001374
1375 cc_library {
1376 name: "libvendorjni",
1377 system_shared_libs: [],
1378 stl: "none",
1379 vendor: true,
1380 }
1381
1382 android_test {
1383 name: "app_vendor",
1384 jni_libs: ["libvendorjni"],
1385 sdk_version: "current",
1386 vendor: true,
1387 }
Colin Cross3c007702020-05-08 11:20:24 -07001388 `)
1389
1390 testCases := []struct {
Colin Crossc2d24052020-05-13 11:05:02 -07001391 name string
1392 sdkJNI bool
1393 vendorJNI bool
Colin Cross3c007702020-05-08 11:20:24 -07001394 }{
Colin Crossc2d24052020-05-13 11:05:02 -07001395 {name: "app_platform"},
1396 {name: "app_sdk", sdkJNI: true},
1397 {name: "app_force_platform"},
1398 {name: "app_force_sdk", sdkJNI: true},
1399 {name: "app_vendor", vendorJNI: true},
Colin Cross3c007702020-05-08 11:20:24 -07001400 }
1401
Colin Crossc2d24052020-05-13 11:05:02 -07001402 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1403 Output("libjni.so").Output.String()
1404 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1405 Output("libjni.so").Output.String()
1406 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1407 Output("libvendorjni.so").Output.String()
1408
Colin Cross3c007702020-05-08 11:20:24 -07001409 for _, test := range testCases {
1410 t.Run(test.name, func(t *testing.T) {
1411 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross3c007702020-05-08 11:20:24 -07001412
1413 jniLibZip := app.MaybeOutput("jnilibs.zip")
1414 if len(jniLibZip.Implicits) != 1 {
1415 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1416 }
1417 gotJNI := jniLibZip.Implicits[0].String()
1418
1419 if test.sdkJNI {
1420 if gotJNI != sdkJNI {
1421 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1422 }
Colin Crossc2d24052020-05-13 11:05:02 -07001423 } else if test.vendorJNI {
1424 if gotJNI != vendorJNI {
1425 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1426 }
Colin Cross3c007702020-05-08 11:20:24 -07001427 } else {
1428 if gotJNI != platformJNI {
1429 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1430 }
1431 }
1432 })
1433 }
1434
1435 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1436 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1437 android_test {
1438 name: "app_platform",
1439 platform_apis: true,
1440 jni_uses_platform_apis: true,
1441 }
1442 `)
1443 })
1444
1445 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1446 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1447 android_test {
1448 name: "app_sdk",
1449 sdk_version: "current",
1450 jni_uses_sdk_apis: true,
1451 }
1452 `)
1453 })
1454
1455}
1456
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001457func TestCertificates(t *testing.T) {
1458 testCases := []struct {
1459 name string
1460 bp string
1461 certificateOverride string
Liz Kammere2b27f42020-05-07 13:24:05 -07001462 expectedLineage string
1463 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001464 }{
1465 {
1466 name: "default",
1467 bp: `
1468 android_app {
1469 name: "foo",
1470 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001471 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001472 }
1473 `,
1474 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001475 expectedLineage: "",
1476 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001477 },
1478 {
1479 name: "module certificate property",
1480 bp: `
1481 android_app {
1482 name: "foo",
1483 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001484 certificate: ":new_certificate",
1485 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001486 }
1487
1488 android_app_certificate {
1489 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001490 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001491 }
1492 `,
1493 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001494 expectedLineage: "",
1495 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001496 },
1497 {
1498 name: "path certificate property",
1499 bp: `
1500 android_app {
1501 name: "foo",
1502 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001503 certificate: "expiredkey",
1504 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001505 }
1506 `,
1507 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001508 expectedLineage: "",
1509 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001510 },
1511 {
1512 name: "certificate overrides",
1513 bp: `
1514 android_app {
1515 name: "foo",
1516 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001517 certificate: "expiredkey",
1518 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001519 }
1520
1521 android_app_certificate {
1522 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001523 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001524 }
1525 `,
1526 certificateOverride: "foo:new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001527 expectedLineage: "",
1528 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1529 },
1530 {
1531 name: "certificate lineage",
1532 bp: `
1533 android_app {
1534 name: "foo",
1535 srcs: ["a.java"],
1536 certificate: ":new_certificate",
1537 lineage: "lineage.bin",
1538 sdk_version: "current",
1539 }
1540
1541 android_app_certificate {
1542 name: "new_certificate",
1543 certificate: "cert/new_cert",
1544 }
1545 `,
1546 certificateOverride: "",
1547 expectedLineage: "--lineage lineage.bin",
1548 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001549 },
1550 }
1551
1552 for _, test := range testCases {
1553 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001554 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001555 if test.certificateOverride != "" {
1556 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1557 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001558 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001559
1560 run(t, ctx, config)
1561 foo := ctx.ModuleForTests("foo", "android_common")
1562
1563 signapk := foo.Output("foo.apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001564 signCertificateFlags := signapk.Args["certificates"]
1565 if test.expectedCertificate != signCertificateFlags {
1566 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1567 }
1568
1569 signFlags := signapk.Args["flags"]
1570 if test.expectedLineage != signFlags {
1571 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001572 }
1573 })
1574 }
1575}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001576
1577func TestPackageNameOverride(t *testing.T) {
1578 testCases := []struct {
1579 name string
1580 bp string
1581 packageNameOverride string
1582 expected []string
1583 }{
1584 {
1585 name: "default",
1586 bp: `
1587 android_app {
1588 name: "foo",
1589 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001590 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001591 }
1592 `,
1593 packageNameOverride: "",
1594 expected: []string{
1595 buildDir + "/.intermediates/foo/android_common/foo.apk",
1596 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1597 },
1598 },
1599 {
1600 name: "overridden",
1601 bp: `
1602 android_app {
1603 name: "foo",
1604 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001605 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001606 }
1607 `,
1608 packageNameOverride: "foo:bar",
1609 expected: []string{
1610 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001611 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001612 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1613 },
1614 },
1615 }
1616
1617 for _, test := range testCases {
1618 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001619 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001620 if test.packageNameOverride != "" {
1621 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1622 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001623 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001624
1625 run(t, ctx, config)
1626 foo := ctx.ModuleForTests("foo", "android_common")
1627
1628 outputs := foo.AllOutputs()
1629 outputMap := make(map[string]bool)
1630 for _, o := range outputs {
1631 outputMap[o] = true
1632 }
1633 for _, e := range test.expected {
1634 if _, exist := outputMap[e]; !exist {
1635 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1636 }
1637 }
1638 })
1639 }
1640}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001641
1642func TestInstrumentationTargetOverridden(t *testing.T) {
1643 bp := `
1644 android_app {
1645 name: "foo",
1646 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001647 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001648 }
1649
1650 android_test {
1651 name: "bar",
1652 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001653 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001654 }
1655 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001656 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001657 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001658 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001659
1660 run(t, ctx, config)
1661
1662 bar := ctx.ModuleForTests("bar", "android_common")
1663 res := bar.Output("package-res.apk")
1664 aapt2Flags := res.Args["flags"]
1665 e := "--rename-instrumentation-target-package org.dandroid.bp"
1666 if !strings.Contains(aapt2Flags, e) {
1667 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1668 }
1669}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001670
1671func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001672 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001673 android_app {
1674 name: "foo",
1675 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001676 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001677 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001678 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001679 }
1680
1681 override_android_app {
1682 name: "bar",
1683 base: "foo",
1684 certificate: ":new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001685 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001686 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001687 }
1688
1689 android_app_certificate {
1690 name: "new_certificate",
1691 certificate: "cert/new_cert",
1692 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001693
1694 override_android_app {
1695 name: "baz",
1696 base: "foo",
1697 package_name: "org.dandroid.bp",
1698 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001699 `)
1700
1701 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001702 moduleName string
1703 variantName string
1704 apkName string
1705 apkPath string
Liz Kammere2b27f42020-05-07 13:24:05 -07001706 certFlag string
1707 lineageFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001708 overrides []string
Liz Kammer1d5983b2020-05-19 19:15:37 +00001709 aaptFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001710 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001711 }{
1712 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001713 moduleName: "foo",
1714 variantName: "android_common",
1715 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001716 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1717 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001718 overrides: []string{"qux"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001719 aaptFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001720 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001721 },
1722 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001723 moduleName: "bar",
1724 variantName: "android_common_bar",
1725 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001726 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1727 lineageFlag: "--lineage lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001728 overrides: []string{"qux", "foo"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001729 aaptFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001730 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001731 },
1732 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001733 moduleName: "baz",
1734 variantName: "android_common_baz",
1735 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001736 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1737 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001738 overrides: []string{"qux", "foo"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001739 aaptFlag: "--rename-manifest-package org.dandroid.bp",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001740 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001741 },
1742 }
1743 for _, expected := range expectedVariants {
1744 variant := ctx.ModuleForTests("foo", expected.variantName)
1745
1746 // Check the final apk name
1747 outputs := variant.AllOutputs()
1748 expectedApkPath := buildDir + expected.apkPath
1749 found := false
1750 for _, o := range outputs {
1751 if o == expectedApkPath {
1752 found = true
1753 break
1754 }
1755 }
1756 if !found {
1757 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1758 }
1759
1760 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001761 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001762 certFlag := signapk.Args["certificates"]
1763 if expected.certFlag != certFlag {
1764 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1765 }
1766
1767 // Check the lineage flags
1768 lineageFlag := signapk.Args["flags"]
1769 if expected.lineageFlag != lineageFlag {
1770 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001771 }
1772
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001773 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001774 mod := variant.Module().(*AndroidApp)
1775 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1776 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1777 expected.overrides, mod.appProperties.Overrides)
1778 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001779
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001780 // Test Overridable property: Logging_parent
1781 logging_parent := mod.aapt.LoggingParent
1782 if expected.logging_parent != logging_parent {
1783 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1784 expected.logging_parent, logging_parent)
1785 }
1786
Liz Kammer1d5983b2020-05-19 19:15:37 +00001787 // Check the package renaming flag, if exists.
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001788 res := variant.Output("package-res.apk")
1789 aapt2Flags := res.Args["flags"]
Liz Kammer1d5983b2020-05-19 19:15:37 +00001790 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1791 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1792 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001793 }
1794}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001795
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001796func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001797 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001798 android_app {
1799 name: "foo",
1800 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001801 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001802 }
1803
1804 override_android_app {
1805 name: "bar",
1806 base: "foo",
1807 package_name: "org.dandroid.bp",
1808 }
1809
1810 android_test {
1811 name: "baz",
1812 srcs: ["b.java"],
1813 instrumentation_for: "foo",
1814 }
1815
1816 android_test {
1817 name: "qux",
1818 srcs: ["b.java"],
1819 instrumentation_for: "bar",
1820 }
1821 `)
1822
1823 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1824 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1825 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1826 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1827 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1828 }
1829
1830 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1831 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1832 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1833 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1834 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1835 }
1836}
1837
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001838func TestOverrideAndroidTest(t *testing.T) {
1839 ctx, _ := testJava(t, `
1840 android_app {
1841 name: "foo",
1842 srcs: ["a.java"],
1843 package_name: "com.android.foo",
1844 sdk_version: "current",
1845 }
1846
1847 override_android_app {
1848 name: "bar",
1849 base: "foo",
1850 package_name: "com.android.bar",
1851 }
1852
1853 android_test {
1854 name: "foo_test",
1855 srcs: ["b.java"],
1856 instrumentation_for: "foo",
1857 }
1858
1859 override_android_test {
1860 name: "bar_test",
1861 base: "foo_test",
1862 package_name: "com.android.bar.test",
1863 instrumentation_for: "bar",
1864 instrumentation_target_package: "com.android.bar",
1865 }
1866 `)
1867
1868 expectedVariants := []struct {
1869 moduleName string
1870 variantName string
1871 apkPath string
1872 overrides []string
1873 targetVariant string
1874 packageFlag string
1875 targetPackageFlag string
1876 }{
1877 {
1878 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001879 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001880 overrides: nil,
1881 targetVariant: "android_common",
1882 packageFlag: "",
1883 targetPackageFlag: "",
1884 },
1885 {
1886 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001887 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001888 overrides: []string{"foo_test"},
1889 targetVariant: "android_common_bar",
1890 packageFlag: "com.android.bar.test",
1891 targetPackageFlag: "com.android.bar",
1892 },
1893 }
1894 for _, expected := range expectedVariants {
1895 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1896
1897 // Check the final apk name
1898 outputs := variant.AllOutputs()
1899 expectedApkPath := buildDir + expected.apkPath
1900 found := false
1901 for _, o := range outputs {
1902 if o == expectedApkPath {
1903 found = true
1904 break
1905 }
1906 }
1907 if !found {
1908 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1909 }
1910
1911 // Check if the overrides field values are correctly aggregated.
1912 mod := variant.Module().(*AndroidTest)
1913 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1914 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1915 expected.overrides, mod.appProperties.Overrides)
1916 }
1917
1918 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1919 javac := variant.Rule("javac")
1920 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1921 if !strings.Contains(javac.Args["classpath"], turbine) {
1922 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1923 }
1924
1925 // Check aapt2 flags.
1926 res := variant.Output("package-res.apk")
1927 aapt2Flags := res.Args["flags"]
1928 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1929 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1930 }
1931}
1932
Jaewoong Jung39982342020-01-14 10:27:18 -08001933func TestAndroidTest_FixTestConfig(t *testing.T) {
1934 ctx, _ := testJava(t, `
1935 android_app {
1936 name: "foo",
1937 srcs: ["a.java"],
1938 package_name: "com.android.foo",
1939 sdk_version: "current",
1940 }
1941
1942 android_test {
1943 name: "foo_test",
1944 srcs: ["b.java"],
1945 instrumentation_for: "foo",
1946 }
1947
1948 android_test {
1949 name: "bar_test",
1950 srcs: ["b.java"],
1951 package_name: "com.android.bar.test",
1952 instrumentation_for: "foo",
1953 }
1954
1955 override_android_test {
1956 name: "baz_test",
1957 base: "foo_test",
1958 package_name: "com.android.baz.test",
1959 }
1960 `)
1961
1962 testCases := []struct {
1963 moduleName string
1964 variantName string
1965 expectedFlags []string
1966 }{
1967 {
1968 moduleName: "foo_test",
1969 variantName: "android_common",
1970 },
1971 {
1972 moduleName: "bar_test",
1973 variantName: "android_common",
1974 expectedFlags: []string{
1975 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
1976 "--package-name com.android.bar.test",
1977 },
1978 },
1979 {
1980 moduleName: "foo_test",
1981 variantName: "android_common_baz_test",
1982 expectedFlags: []string{
1983 "--manifest " + buildDir +
1984 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
1985 "--package-name com.android.baz.test",
1986 "--test-file-name baz_test.apk",
1987 },
1988 },
1989 }
1990
1991 for _, test := range testCases {
1992 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
1993 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
1994
1995 if len(test.expectedFlags) > 0 {
1996 if params.Rule == nil {
1997 t.Errorf("test_config_fixer was expected to run, but didn't")
1998 } else {
1999 for _, flag := range test.expectedFlags {
2000 if !strings.Contains(params.RuleParams.Command, flag) {
2001 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2002 }
2003 }
2004 }
2005 } else {
2006 if params.Rule != nil {
2007 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2008 }
2009 }
2010
2011 }
2012}
2013
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002014func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002015 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002016 android_app_import {
2017 name: "foo",
2018 apk: "prebuilts/apk/app.apk",
2019 certificate: "platform",
2020 dex_preopt: {
2021 enabled: true,
2022 },
2023 }
2024 `)
2025
2026 variant := ctx.ModuleForTests("foo", "android_common")
2027
2028 // Check dexpreopt outputs.
2029 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2030 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2031 t.Errorf("can't find dexpreopt outputs")
2032 }
2033
2034 // Check cert signing flag.
2035 signedApk := variant.Output("signed/foo.apk")
2036 signingFlag := signedApk.Args["certificates"]
2037 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2038 if expected != signingFlag {
2039 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2040 }
2041}
2042
2043func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002044 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002045 android_app_import {
2046 name: "foo",
2047 apk: "prebuilts/apk/app.apk",
2048 certificate: "platform",
2049 dex_preopt: {
2050 enabled: false,
2051 },
2052 }
2053 `)
2054
2055 variant := ctx.ModuleForTests("foo", "android_common")
2056
2057 // Check dexpreopt outputs. They shouldn't exist.
2058 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2059 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2060 t.Errorf("dexpreopt shouldn't have run.")
2061 }
2062}
2063
2064func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002065 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002066 android_app_import {
2067 name: "foo",
2068 apk: "prebuilts/apk/app.apk",
2069 presigned: true,
2070 dex_preopt: {
2071 enabled: true,
2072 },
2073 }
2074 `)
2075
2076 variant := ctx.ModuleForTests("foo", "android_common")
2077
2078 // Check dexpreopt outputs.
2079 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2080 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2081 t.Errorf("can't find dexpreopt outputs")
2082 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002083 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002084 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2085 t.Errorf("signing rule shouldn't be included.")
2086 }
2087 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2088 t.Errorf("can't find aligning rule")
2089 }
2090}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002091
Liz Kammer24978992020-05-13 15:49:21 -07002092func TestAndroidAppImport_SigningLineage(t *testing.T) {
2093 ctx, _ := testJava(t, `
2094 android_app_import {
2095 name: "foo",
2096 apk: "prebuilts/apk/app.apk",
2097 certificate: "platform",
2098 lineage: "lineage.bin",
2099 }
2100 `)
2101
2102 variant := ctx.ModuleForTests("foo", "android_common")
2103
2104 // Check cert signing lineage flag.
2105 signedApk := variant.Output("signed/foo.apk")
2106 signingFlag := signedApk.Args["flags"]
2107 expected := "--lineage lineage.bin"
2108 if expected != signingFlag {
2109 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2110 }
2111}
2112
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002113func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2114 ctx, _ := testJava(t, `
2115 android_app_import {
2116 name: "foo",
2117 apk: "prebuilts/apk/app.apk",
2118 default_dev_cert: true,
2119 dex_preopt: {
2120 enabled: true,
2121 },
2122 }
2123 `)
2124
2125 variant := ctx.ModuleForTests("foo", "android_common")
2126
2127 // Check dexpreopt outputs.
2128 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2129 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2130 t.Errorf("can't find dexpreopt outputs")
2131 }
2132
2133 // Check cert signing flag.
2134 signedApk := variant.Output("signed/foo.apk")
2135 signingFlag := signedApk.Args["certificates"]
2136 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2137 if expected != signingFlag {
2138 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2139 }
2140}
2141
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002142func TestAndroidAppImport_DpiVariants(t *testing.T) {
2143 bp := `
2144 android_app_import {
2145 name: "foo",
2146 apk: "prebuilts/apk/app.apk",
2147 dpi_variants: {
2148 xhdpi: {
2149 apk: "prebuilts/apk/app_xhdpi.apk",
2150 },
2151 xxhdpi: {
2152 apk: "prebuilts/apk/app_xxhdpi.apk",
2153 },
2154 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002155 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002156 dex_preopt: {
2157 enabled: true,
2158 },
2159 }
2160 `
2161 testCases := []struct {
2162 name string
2163 aaptPreferredConfig *string
2164 aaptPrebuiltDPI []string
2165 expected string
2166 }{
2167 {
2168 name: "no preferred",
2169 aaptPreferredConfig: nil,
2170 aaptPrebuiltDPI: []string{},
2171 expected: "prebuilts/apk/app.apk",
2172 },
2173 {
2174 name: "AAPTPreferredConfig matches",
2175 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002176 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002177 expected: "prebuilts/apk/app_xhdpi.apk",
2178 },
2179 {
2180 name: "AAPTPrebuiltDPI matches",
2181 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2182 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2183 expected: "prebuilts/apk/app_xxhdpi.apk",
2184 },
2185 {
2186 name: "non-first AAPTPrebuiltDPI matches",
2187 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2188 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2189 expected: "prebuilts/apk/app_xhdpi.apk",
2190 },
2191 {
2192 name: "no matches",
2193 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2194 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2195 expected: "prebuilts/apk/app.apk",
2196 },
2197 }
2198
2199 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2200 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002201 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002202 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2203 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002204 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002205
2206 run(t, ctx, config)
2207
2208 variant := ctx.ModuleForTests("foo", "android_common")
2209 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2210 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2211 if len(matches) != 2 {
2212 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2213 }
2214 if test.expected != matches[1] {
2215 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2216 }
2217 }
2218}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002219
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002220func TestAndroidAppImport_Filename(t *testing.T) {
2221 ctx, config := testJava(t, `
2222 android_app_import {
2223 name: "foo",
2224 apk: "prebuilts/apk/app.apk",
2225 presigned: true,
2226 }
2227
2228 android_app_import {
2229 name: "bar",
2230 apk: "prebuilts/apk/app.apk",
2231 presigned: true,
2232 filename: "bar_sample.apk"
2233 }
2234 `)
2235
2236 testCases := []struct {
2237 name string
2238 expected string
2239 }{
2240 {
2241 name: "foo",
2242 expected: "foo.apk",
2243 },
2244 {
2245 name: "bar",
2246 expected: "bar_sample.apk",
2247 },
2248 }
2249
2250 for _, test := range testCases {
2251 variant := ctx.ModuleForTests(test.name, "android_common")
2252 if variant.MaybeOutput(test.expected).Rule == nil {
2253 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2254 }
2255
2256 a := variant.Module().(*AndroidAppImport)
2257 expectedValues := []string{test.expected}
2258 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002259 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002260 if !reflect.DeepEqual(actualValues, expectedValues) {
2261 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2262 actualValues, expectedValues)
2263 }
2264 }
2265}
2266
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002267func TestAndroidAppImport_ArchVariants(t *testing.T) {
2268 // The test config's target arch is ARM64.
2269 testCases := []struct {
2270 name string
2271 bp string
2272 expected string
2273 }{
2274 {
2275 name: "matching arch",
2276 bp: `
2277 android_app_import {
2278 name: "foo",
2279 apk: "prebuilts/apk/app.apk",
2280 arch: {
2281 arm64: {
2282 apk: "prebuilts/apk/app_arm64.apk",
2283 },
2284 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002285 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002286 dex_preopt: {
2287 enabled: true,
2288 },
2289 }
2290 `,
2291 expected: "prebuilts/apk/app_arm64.apk",
2292 },
2293 {
2294 name: "no matching arch",
2295 bp: `
2296 android_app_import {
2297 name: "foo",
2298 apk: "prebuilts/apk/app.apk",
2299 arch: {
2300 arm: {
2301 apk: "prebuilts/apk/app_arm.apk",
2302 },
2303 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002304 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002305 dex_preopt: {
2306 enabled: true,
2307 },
2308 }
2309 `,
2310 expected: "prebuilts/apk/app.apk",
2311 },
2312 }
2313
2314 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2315 for _, test := range testCases {
2316 ctx, _ := testJava(t, test.bp)
2317
2318 variant := ctx.ModuleForTests("foo", "android_common")
2319 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2320 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2321 if len(matches) != 2 {
2322 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2323 }
2324 if test.expected != matches[1] {
2325 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2326 }
2327 }
2328}
2329
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002330func TestAndroidTestImport(t *testing.T) {
2331 ctx, config := testJava(t, `
2332 android_test_import {
2333 name: "foo",
2334 apk: "prebuilts/apk/app.apk",
2335 presigned: true,
2336 data: [
2337 "testdata/data",
2338 ],
2339 }
2340 `)
2341
2342 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2343
2344 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002345 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002346 expected := []string{"tests"}
2347 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2348 if !reflect.DeepEqual(expected, actual) {
2349 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2350 }
2351 expected = []string{"testdata/data:testdata/data"}
2352 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2353 if !reflect.DeepEqual(expected, actual) {
2354 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2355 }
2356}
2357
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002358func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2359 ctx, _ := testJava(t, `
2360 android_test_import {
2361 name: "foo",
2362 apk: "prebuilts/apk/app.apk",
2363 certificate: "cert/new_cert",
2364 data: [
2365 "testdata/data",
2366 ],
2367 }
2368
2369 android_test_import {
2370 name: "foo_presigned",
2371 apk: "prebuilts/apk/app.apk",
2372 presigned: true,
2373 data: [
2374 "testdata/data",
2375 ],
2376 }
2377 `)
2378
2379 variant := ctx.ModuleForTests("foo", "android_common")
2380 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2381 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2382 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2383 }
2384
2385 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2386 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2387 if jniRule != android.Cp.String() {
2388 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2389 }
Liz Kammer3b70b3f2020-05-20 14:36:30 -07002390 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2391 t.Errorf("Presigned test apk should be aligned")
2392 }
2393}
2394
2395func TestAndroidTestImport_Preprocessed(t *testing.T) {
2396 ctx, _ := testJava(t, `
2397 android_test_import {
2398 name: "foo",
2399 apk: "prebuilts/apk/app.apk",
2400 presigned: true,
2401 preprocessed: true,
2402 }
2403
2404 android_test_import {
2405 name: "foo_cert",
2406 apk: "prebuilts/apk/app.apk",
2407 certificate: "cert/new_cert",
2408 preprocessed: true,
2409 }
2410 `)
2411
2412 testModules := []string{"foo", "foo_cert"}
2413 for _, m := range testModules {
2414 apkName := m + ".apk"
2415 variant := ctx.ModuleForTests(m, "android_common")
2416 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2417 if jniRule != android.Cp.String() {
2418 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2419 }
2420
2421 // Make sure signing and aligning were skipped.
2422 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2423 t.Errorf("signing rule shouldn't be included for preprocessed.")
2424 }
2425 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2426 t.Errorf("aligning rule shouldn't be for preprocessed")
2427 }
2428 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002429}
2430
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002431func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002432 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002433 cc_library {
2434 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002435 sdk_version: "current",
2436 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002437 }
2438
2439 android_test {
2440 name: "stl",
2441 jni_libs: ["libjni"],
2442 compile_multilib: "both",
2443 sdk_version: "current",
2444 stl: "c++_shared",
2445 }
2446
2447 android_test {
2448 name: "system",
2449 jni_libs: ["libjni"],
2450 compile_multilib: "both",
2451 sdk_version: "current",
2452 }
2453 `)
2454
2455 testCases := []struct {
2456 name string
2457 jnis []string
2458 }{
2459 {"stl",
2460 []string{
2461 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002462 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002463 },
2464 },
2465 {"system",
2466 []string{
2467 "libjni.so",
2468 },
2469 },
2470 }
2471
2472 for _, test := range testCases {
2473 t.Run(test.name, func(t *testing.T) {
2474 app := ctx.ModuleForTests(test.name, "android_common")
2475 jniLibZip := app.Output("jnilibs.zip")
2476 var jnis []string
2477 args := strings.Fields(jniLibZip.Args["jarArgs"])
2478 for i := 0; i < len(args); i++ {
2479 if args[i] == "-f" {
2480 jnis = append(jnis, args[i+1])
2481 i += 1
2482 }
2483 }
2484 jnisJoined := strings.Join(jnis, " ")
2485 for _, jni := range test.jnis {
2486 if !strings.Contains(jnisJoined, jni) {
2487 t.Errorf("missing jni %q in %q", jni, jnis)
2488 }
2489 }
2490 })
2491 }
2492}
Colin Cross50ddcc42019-05-16 12:28:22 -07002493
2494func TestUsesLibraries(t *testing.T) {
2495 bp := `
2496 java_sdk_library {
2497 name: "foo",
2498 srcs: ["a.java"],
2499 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002500 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002501 }
2502
2503 java_sdk_library {
Paul Duffin859fe962020-05-15 10:20:31 +01002504 name: "qux",
2505 srcs: ["a.java"],
2506 api_packages: ["qux"],
2507 sdk_version: "current",
2508 }
2509
2510 java_sdk_library {
2511 name: "quuz",
2512 srcs: ["a.java"],
2513 api_packages: ["quuz"],
2514 sdk_version: "current",
2515 }
2516
2517 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002518 name: "bar",
2519 srcs: ["a.java"],
2520 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002521 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002522 }
2523
2524 android_app {
2525 name: "app",
2526 srcs: ["a.java"],
Paul Duffin859fe962020-05-15 10:20:31 +01002527 libs: ["qux", "quuz.stubs"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002528 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002529 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002530 optional_uses_libs: [
2531 "bar",
2532 "baz",
2533 ],
2534 }
2535
2536 android_app_import {
2537 name: "prebuilt",
2538 apk: "prebuilts/apk/app.apk",
2539 certificate: "platform",
2540 uses_libs: ["foo"],
2541 optional_uses_libs: [
2542 "bar",
2543 "baz",
2544 ],
2545 }
2546 `
2547
Colin Cross98be1bb2019-12-13 20:41:13 -08002548 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002549 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2550
Colin Cross98be1bb2019-12-13 20:41:13 -08002551 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002552
2553 run(t, ctx, config)
2554
2555 app := ctx.ModuleForTests("app", "android_common")
2556 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2557
Paul Duffin859fe962020-05-15 10:20:31 +01002558 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2559 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2560 if w := "--uses-library qux"; !strings.Contains(manifestFixerArgs, w) {
2561 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2562 }
2563 if w := "--uses-library quuz"; !strings.Contains(manifestFixerArgs, w) {
2564 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2565 }
2566
Colin Cross50ddcc42019-05-16 12:28:22 -07002567 // Test that all libraries are verified
2568 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2569 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2570 t.Errorf("wanted %q in %q", w, cmd)
2571 }
2572
2573 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2574 t.Errorf("wanted %q in %q", w, cmd)
2575 }
2576
2577 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2578
2579 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2580 t.Errorf("wanted %q in %q", w, cmd)
2581 }
2582
2583 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2584 t.Errorf("wanted %q in %q", w, cmd)
2585 }
2586
2587 // Test that only present libraries are preopted
2588 cmd = app.Rule("dexpreopt").RuleParams.Command
2589
Ulya Trafimovich5f364b62020-06-30 12:39:01 +01002590 if w := `--target-classpath-for-sdk any /system/framework/foo.jar:/system/framework/bar.jar`; !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002591 t.Errorf("wanted %q in %q", w, cmd)
2592 }
2593
2594 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2595
Ulya Trafimovich5f364b62020-06-30 12:39:01 +01002596 if w := `--target-classpath-for-sdk any /system/framework/foo.jar:/system/framework/bar.jar`; !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002597 t.Errorf("wanted %q in %q", w, cmd)
2598 }
2599}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002600
2601func TestCodelessApp(t *testing.T) {
2602 testCases := []struct {
2603 name string
2604 bp string
2605 noCode bool
2606 }{
2607 {
2608 name: "normal",
2609 bp: `
2610 android_app {
2611 name: "foo",
2612 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002613 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002614 }
2615 `,
2616 noCode: false,
2617 },
2618 {
2619 name: "app without sources",
2620 bp: `
2621 android_app {
2622 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002623 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002624 }
2625 `,
2626 noCode: true,
2627 },
2628 {
2629 name: "app with libraries",
2630 bp: `
2631 android_app {
2632 name: "foo",
2633 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002634 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002635 }
2636
2637 java_library {
2638 name: "lib",
2639 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002640 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002641 }
2642 `,
2643 noCode: false,
2644 },
2645 {
2646 name: "app with sourceless libraries",
2647 bp: `
2648 android_app {
2649 name: "foo",
2650 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002651 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002652 }
2653
2654 java_library {
2655 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002656 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002657 }
2658 `,
2659 // TODO(jungjw): this should probably be true
2660 noCode: false,
2661 },
2662 }
2663
2664 for _, test := range testCases {
2665 t.Run(test.name, func(t *testing.T) {
2666 ctx := testApp(t, test.bp)
2667
2668 foo := ctx.ModuleForTests("foo", "android_common")
2669 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2670 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2671 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2672 }
2673 })
2674 }
2675}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002676
2677func TestEmbedNotice(t *testing.T) {
Colin Cross238c1f32020-06-07 16:58:18 -07002678 ctx, _ := testJavaWithFS(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002679 android_app {
2680 name: "foo",
2681 srcs: ["a.java"],
2682 static_libs: ["javalib"],
2683 jni_libs: ["libjni"],
2684 notice: "APP_NOTICE",
2685 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002686 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002687 }
2688
2689 // No embed_notice flag
2690 android_app {
2691 name: "bar",
2692 srcs: ["a.java"],
2693 jni_libs: ["libjni"],
2694 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002695 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002696 }
2697
2698 // No NOTICE files
2699 android_app {
2700 name: "baz",
2701 srcs: ["a.java"],
2702 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002703 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002704 }
2705
2706 cc_library {
2707 name: "libjni",
2708 system_shared_libs: [],
2709 stl: "none",
2710 notice: "LIB_NOTICE",
Colin Cross094cde42020-02-15 10:38:00 -08002711 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002712 }
2713
2714 java_library {
2715 name: "javalib",
2716 srcs: [
2717 ":gen",
2718 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002719 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002720 }
2721
2722 genrule {
2723 name: "gen",
2724 tools: ["gentool"],
2725 out: ["gen.java"],
2726 notice: "GENRULE_NOTICE",
2727 }
2728
2729 java_binary_host {
2730 name: "gentool",
2731 srcs: ["b.java"],
2732 notice: "TOOL_NOTICE",
2733 }
Colin Cross238c1f32020-06-07 16:58:18 -07002734 `, map[string][]byte{
2735 "APP_NOTICE": nil,
2736 "GENRULE_NOTICE": nil,
2737 "LIB_NOTICE": nil,
2738 "TOOL_NOTICE": nil,
2739 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002740
2741 // foo has NOTICE files to process, and embed_notices is true.
2742 foo := ctx.ModuleForTests("foo", "android_common")
2743 // verify merge notices rule.
2744 mergeNotices := foo.Rule("mergeNoticesRule")
2745 noticeInputs := mergeNotices.Inputs.Strings()
2746 // TOOL_NOTICE should be excluded as it's a host module.
2747 if len(mergeNotices.Inputs) != 3 {
2748 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2749 }
2750 if !inList("APP_NOTICE", noticeInputs) {
2751 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2752 }
2753 if !inList("LIB_NOTICE", noticeInputs) {
2754 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2755 }
2756 if !inList("GENRULE_NOTICE", noticeInputs) {
2757 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2758 }
2759 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2760 res := foo.Output("package-res.apk")
2761 aapt2Flags := res.Args["flags"]
2762 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2763 if !strings.Contains(aapt2Flags, e) {
2764 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2765 }
2766
2767 // bar has NOTICE files to process, but embed_notices is not set.
2768 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002769 res = bar.Output("package-res.apk")
2770 aapt2Flags = res.Args["flags"]
2771 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2772 if strings.Contains(aapt2Flags, e) {
2773 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002774 }
2775
2776 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2777 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002778 res = baz.Output("package-res.apk")
2779 aapt2Flags = res.Args["flags"]
2780 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2781 if strings.Contains(aapt2Flags, e) {
2782 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002783 }
2784}
Colin Cross53a87f52019-06-25 13:35:30 -07002785
2786func TestUncompressDex(t *testing.T) {
2787 testCases := []struct {
2788 name string
2789 bp string
2790
2791 uncompressedPlatform bool
2792 uncompressedUnbundled bool
2793 }{
2794 {
2795 name: "normal",
2796 bp: `
2797 android_app {
2798 name: "foo",
2799 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002800 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002801 }
2802 `,
2803 uncompressedPlatform: true,
2804 uncompressedUnbundled: false,
2805 },
2806 {
2807 name: "use_embedded_dex",
2808 bp: `
2809 android_app {
2810 name: "foo",
2811 use_embedded_dex: true,
2812 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002813 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002814 }
2815 `,
2816 uncompressedPlatform: true,
2817 uncompressedUnbundled: true,
2818 },
2819 {
2820 name: "privileged",
2821 bp: `
2822 android_app {
2823 name: "foo",
2824 privileged: true,
2825 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002826 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002827 }
2828 `,
2829 uncompressedPlatform: true,
2830 uncompressedUnbundled: true,
2831 },
David Srbeckye033cba2020-05-20 22:20:28 +01002832 {
2833 name: "normal_uncompress_dex_true",
2834 bp: `
2835 android_app {
2836 name: "foo",
2837 srcs: ["a.java"],
2838 sdk_version: "current",
2839 uncompress_dex: true,
2840 }
2841 `,
2842 uncompressedPlatform: true,
2843 uncompressedUnbundled: true,
2844 },
2845 {
2846 name: "normal_uncompress_dex_false",
2847 bp: `
2848 android_app {
2849 name: "foo",
2850 srcs: ["a.java"],
2851 sdk_version: "current",
2852 uncompress_dex: false,
2853 }
2854 `,
2855 uncompressedPlatform: false,
2856 uncompressedUnbundled: false,
2857 },
Colin Cross53a87f52019-06-25 13:35:30 -07002858 }
2859
2860 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2861 t.Helper()
2862
Colin Cross98be1bb2019-12-13 20:41:13 -08002863 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002864 if unbundled {
2865 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2866 }
2867
Colin Cross98be1bb2019-12-13 20:41:13 -08002868 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002869
2870 run(t, ctx, config)
2871
2872 foo := ctx.ModuleForTests("foo", "android_common")
2873 dex := foo.Rule("r8")
2874 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2875 aligned := foo.MaybeRule("zipalign").Rule != nil
2876
2877 if uncompressedInDexJar != want {
2878 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2879 }
2880
2881 if aligned != want {
2882 t.Errorf("want aligned %v, got %v", want, aligned)
2883 }
2884 }
2885
2886 for _, tt := range testCases {
2887 t.Run(tt.name, func(t *testing.T) {
2888 t.Run("platform", func(t *testing.T) {
2889 test(t, tt.bp, tt.uncompressedPlatform, false)
2890 })
2891 t.Run("unbundled", func(t *testing.T) {
2892 test(t, tt.bp, tt.uncompressedUnbundled, true)
2893 })
2894 })
2895 }
2896}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002897
2898func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2899 if expectedValue != "" {
2900 expectedFlag := "--" + flagName + " " + expectedValue
2901 if !strings.Contains(aapt2Flags, expectedFlag) {
2902 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2903 }
2904 } else {
2905 unexpectedFlag := "--" + flagName
2906 if strings.Contains(aapt2Flags, unexpectedFlag) {
2907 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2908 }
2909 }
2910}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002911
2912func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002913 fs := map[string][]byte{
2914 "baz/res/res/values/strings.xml": nil,
2915 "bar/res/res/values/strings.xml": nil,
2916 }
2917 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002918 runtime_resource_overlay {
2919 name: "foo",
2920 certificate: "platform",
Liz Kammer966b2f02020-05-19 16:15:25 -07002921 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002922 product_specific: true,
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002923 static_libs: ["bar"],
2924 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002925 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002926 }
2927
2928 runtime_resource_overlay {
2929 name: "foo_themed",
2930 certificate: "platform",
2931 product_specific: true,
2932 theme: "faza",
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002933 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002934 }
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002935
2936 android_library {
2937 name: "bar",
2938 resource_dirs: ["bar/res"],
2939 }
2940
2941 android_app {
2942 name: "baz",
2943 sdk_version: "current",
2944 resource_dirs: ["baz/res"],
2945 }
2946 `
2947 config := testAppConfig(nil, bp, fs)
2948 ctx := testContext()
2949 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002950
2951 m := ctx.ModuleForTests("foo", "android_common")
2952
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002953 // Check AAPT2 link flags.
2954 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2955 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2956 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2957 if len(absentFlags) > 0 {
2958 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2959 }
2960
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002961 // Check overlay.list output for static_libs dependency.
2962 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
2963 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
2964 if !inList(staticLibPackage, overlayList) {
2965 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
2966 }
2967
2968 // Check AAPT2 link flags for resource_libs dependency.
2969 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
2970 if !strings.Contains(aapt2Flags, resourceLibFlag) {
2971 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
2972 }
2973
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002974 // Check cert signing flag.
2975 signedApk := m.Output("signed/foo.apk")
Liz Kammer966b2f02020-05-19 16:15:25 -07002976 lineageFlag := signedApk.Args["flags"]
2977 expectedLineageFlag := "--lineage lineage.bin"
2978 if expectedLineageFlag != lineageFlag {
2979 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
2980 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002981 signingFlag := signedApk.Args["certificates"]
2982 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2983 if expected != signingFlag {
2984 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2985 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002986 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2987 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002988 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
2989 if !reflect.DeepEqual(path, expectedPath) {
2990 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
2991 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002992
2993 // Check device location.
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002994 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002995 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002996 if !reflect.DeepEqual(path, expectedPath) {
2997 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2998 }
2999
3000 // A themed module has a different device location
3001 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003002 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3003 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003004 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3005 if !reflect.DeepEqual(path, expectedPath) {
3006 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3007 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003008
3009 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3010 expectedOverrides := []string{"foo"}
3011 if !reflect.DeepEqual(overrides, expectedOverrides) {
3012 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3013 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003014}
Roshan Pius4df2bc72020-04-27 09:42:27 -07003015
3016func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3017 ctx, _ := testJava(t, `
3018 runtime_resource_overlay {
3019 name: "foo_overlay",
3020 certificate: "platform",
3021 product_specific: true,
3022 sdk_version: "current",
3023 }
3024
3025 override_runtime_resource_overlay {
3026 name: "bar_overlay",
3027 base: "foo_overlay",
3028 package_name: "com.android.bar.overlay",
3029 target_package_name: "com.android.bar",
3030 }
3031 `)
3032
3033 expectedVariants := []struct {
3034 moduleName string
3035 variantName string
3036 apkPath string
3037 overrides []string
3038 targetVariant string
3039 packageFlag string
3040 targetPackageFlag string
3041 }{
3042 {
3043 variantName: "android_common",
3044 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3045 overrides: nil,
3046 targetVariant: "android_common",
3047 packageFlag: "",
3048 targetPackageFlag: "",
3049 },
3050 {
3051 variantName: "android_common_bar_overlay",
3052 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3053 overrides: []string{"foo_overlay"},
3054 targetVariant: "android_common_bar",
3055 packageFlag: "com.android.bar.overlay",
3056 targetPackageFlag: "com.android.bar",
3057 },
3058 }
3059 for _, expected := range expectedVariants {
3060 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3061
3062 // Check the final apk name
3063 outputs := variant.AllOutputs()
3064 expectedApkPath := buildDir + expected.apkPath
3065 found := false
3066 for _, o := range outputs {
3067 if o == expectedApkPath {
3068 found = true
3069 break
3070 }
3071 }
3072 if !found {
3073 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3074 }
3075
3076 // Check if the overrides field values are correctly aggregated.
3077 mod := variant.Module().(*RuntimeResourceOverlay)
3078 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3079 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3080 expected.overrides, mod.properties.Overrides)
3081 }
3082
3083 // Check aapt2 flags.
3084 res := variant.Output("package-res.apk")
3085 aapt2Flags := res.Args["flags"]
3086 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
3087 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3088 }
3089}
Jaewoong Jungbf135462020-04-26 15:10:51 -07003090
3091func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3092 ctx, config := testJava(t, `
3093 java_defaults {
3094 name: "rro_defaults",
3095 theme: "default_theme",
3096 product_specific: true,
3097 aaptflags: ["--keep-raw-values"],
3098 }
3099
3100 runtime_resource_overlay {
3101 name: "foo_with_defaults",
3102 defaults: ["rro_defaults"],
3103 }
3104
3105 runtime_resource_overlay {
3106 name: "foo_barebones",
3107 }
3108 `)
3109
3110 //
3111 // RRO module with defaults
3112 //
3113 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3114
3115 // Check AAPT2 link flags.
3116 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3117 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3118 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3119 if len(absentFlags) > 0 {
3120 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3121 }
3122
3123 // Check device location.
3124 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3125 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3126 if !reflect.DeepEqual(path, expectedPath) {
3127 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3128 }
3129
3130 //
3131 // RRO module without defaults
3132 //
3133 m = ctx.ModuleForTests("foo_barebones", "android_common")
3134
3135 // Check AAPT2 link flags.
3136 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3137 unexpectedFlags := "--keep-raw-values"
3138 if inList(unexpectedFlags, aapt2Flags) {
3139 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3140 }
3141
3142 // Check device location.
3143 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3144 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3145 if !reflect.DeepEqual(path, expectedPath) {
3146 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3147 }
3148}