blob: 45dab6202b8d7806206eb01fb96e68d83be10b25 [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
Jeongik Cha538c0d02019-07-11 15:54:27 +0900144func TestPlatformAPIs(t *testing.T) {
145 testJava(t, `
146 android_app {
147 name: "foo",
148 srcs: ["a.java"],
149 platform_apis: true,
150 }
151 `)
152
153 testJava(t, `
154 android_app {
155 name: "foo",
156 srcs: ["a.java"],
157 sdk_version: "current",
158 }
159 `)
160
161 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
162 android_app {
163 name: "bar",
164 srcs: ["b.java"],
165 }
166 `)
167
168 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
169 android_app {
170 name: "bar",
171 srcs: ["b.java"],
172 sdk_version: "system_current",
173 platform_apis: true,
174 }
175 `)
176}
177
Jeongik Chae403e9e2019-12-07 00:16:24 +0900178func TestAndroidAppLinkType(t *testing.T) {
179 testJava(t, `
180 android_app {
181 name: "foo",
182 srcs: ["a.java"],
183 libs: ["bar"],
184 static_libs: ["baz"],
185 platform_apis: true,
186 }
187
188 java_library {
189 name: "bar",
190 sdk_version: "current",
191 srcs: ["b.java"],
192 }
193
194 android_library {
195 name: "baz",
196 sdk_version: "system_current",
197 srcs: ["c.java"],
198 }
199 `)
200
201 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.", `
202 android_app {
203 name: "foo",
204 srcs: ["a.java"],
205 libs: ["bar"],
206 sdk_version: "current",
207 static_libs: ["baz"],
208 }
209
210 java_library {
211 name: "bar",
212 sdk_version: "current",
213 srcs: ["b.java"],
214 }
215
216 android_library {
217 name: "baz",
218 sdk_version: "system_current",
219 srcs: ["c.java"],
220 }
221 `)
222
223 testJava(t, `
224 android_app {
225 name: "foo",
226 srcs: ["a.java"],
227 libs: ["bar"],
228 sdk_version: "system_current",
229 static_libs: ["baz"],
230 }
231
232 java_library {
233 name: "bar",
234 sdk_version: "current",
235 srcs: ["b.java"],
236 }
237
238 android_library {
239 name: "baz",
240 sdk_version: "system_current",
241 srcs: ["c.java"],
242 }
243 `)
244
245 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.", `
246 android_app {
247 name: "foo",
248 srcs: ["a.java"],
249 libs: ["bar"],
250 sdk_version: "system_current",
251 static_libs: ["baz"],
252 }
253
254 java_library {
255 name: "bar",
256 sdk_version: "current",
257 srcs: ["b.java"],
258 }
259
260 android_library {
261 name: "baz",
262 srcs: ["c.java"],
263 }
264 `)
265}
266
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100267func TestUpdatableApps(t *testing.T) {
268 testCases := []struct {
269 name string
270 bp string
271 expectedError string
272 }{
273 {
274 name: "Stable public SDK",
275 bp: `android_app {
276 name: "foo",
277 srcs: ["a.java"],
278 sdk_version: "29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100279 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100280 updatable: true,
281 }`,
282 },
283 {
284 name: "Stable system SDK",
285 bp: `android_app {
286 name: "foo",
287 srcs: ["a.java"],
288 sdk_version: "system_29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100289 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100290 updatable: true,
291 }`,
292 },
293 {
294 name: "Current public SDK",
295 bp: `android_app {
296 name: "foo",
297 srcs: ["a.java"],
298 sdk_version: "current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100299 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100300 updatable: true,
301 }`,
302 },
303 {
304 name: "Current system SDK",
305 bp: `android_app {
306 name: "foo",
307 srcs: ["a.java"],
308 sdk_version: "system_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100309 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100310 updatable: true,
311 }`,
312 },
313 {
314 name: "Current module SDK",
315 bp: `android_app {
316 name: "foo",
317 srcs: ["a.java"],
318 sdk_version: "module_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100319 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100320 updatable: true,
321 }`,
322 },
323 {
324 name: "Current core SDK",
325 bp: `android_app {
326 name: "foo",
327 srcs: ["a.java"],
328 sdk_version: "core_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100329 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100330 updatable: true,
331 }`,
332 },
333 {
334 name: "No Platform APIs",
335 bp: `android_app {
336 name: "foo",
337 srcs: ["a.java"],
338 platform_apis: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100339 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100340 updatable: true,
341 }`,
342 expectedError: "Updatable apps must use stable SDKs",
343 },
344 {
345 name: "No Core Platform APIs",
346 bp: `android_app {
347 name: "foo",
348 srcs: ["a.java"],
349 sdk_version: "core_platform",
Artur Satayevf40fc852020-04-16 13:43:02 +0100350 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100351 updatable: true,
352 }`,
353 expectedError: "Updatable apps must use stable SDKs",
354 },
355 {
356 name: "No unspecified APIs",
357 bp: `android_app {
358 name: "foo",
359 srcs: ["a.java"],
360 updatable: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100361 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100362 }`,
363 expectedError: "Updatable apps must use stable SDK",
364 },
Artur Satayevf40fc852020-04-16 13:43:02 +0100365 {
366 name: "Must specify min_sdk_version",
367 bp: `android_app {
368 name: "app_without_min_sdk_version",
369 srcs: ["a.java"],
370 sdk_version: "29",
371 updatable: true,
372 }`,
373 expectedError: "updatable apps must set min_sdk_version.",
374 },
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100375 }
376
377 for _, test := range testCases {
378 t.Run(test.name, func(t *testing.T) {
379 if test.expectedError == "" {
380 testJava(t, test.bp)
381 } else {
382 testJavaError(t, test.expectedError, test.bp)
383 }
384 })
385 }
386}
387
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900388func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
389 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
390 android_app {
391 name: "foo",
392 srcs: ["a.java"],
393 updatable: true,
394 sdk_version: "current",
395 min_sdk_version: "current",
396 jni_libs: ["libjni"],
397 }
398
399 cc_library {
400 name: "libjni",
401 stl: "none",
402 system_shared_libs: [],
403 sdk_version: "current",
404 }
405 `)
406}
407
408func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
409 bp := cc.GatherRequiredDepsForTest(android.Android) + `
410 android_app {
411 name: "foo",
412 srcs: ["a.java"],
413 updatable: true,
414 sdk_version: "current",
415 min_sdk_version: "29",
416 jni_libs: ["libjni"],
417 }
418
419 cc_library {
420 name: "libjni",
421 stl: "none",
422 system_shared_libs: [],
423 sdk_version: "29",
424 }
425
426 ndk_prebuilt_object {
427 name: "ndk_crtbegin_so.29",
428 sdk_version: "29",
429 }
430
431 ndk_prebuilt_object {
432 name: "ndk_crtend_so.29",
433 sdk_version: "29",
434 }
435 `
436 fs := map[string][]byte{
437 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
438 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
439 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
440 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
441 }
442
443 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
444
445 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
446 var crtbeginFound, crtendFound bool
447 for _, input := range inputs {
448 switch input.String() {
449 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o":
450 crtbeginFound = true
451 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o":
452 crtendFound = true
453 }
454 }
455 if !crtbeginFound || !crtendFound {
456 t.Error("should link with ndk_crtbegin_so.29 and ndk_crtend_so.29")
457 }
458}
459
460func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
461 bp := cc.GatherRequiredDepsForTest(android.Android) + `
462 android_app {
463 name: "foo",
464 srcs: ["a.java"],
465 updatable: true,
466 sdk_version: "current",
467 min_sdk_version: "29", // this APK should support 29
468 jni_libs: ["libjni"],
469 }
470
471 cc_library {
472 name: "libjni",
473 stl: "none",
474 sdk_version: "current",
475 }
476 `
477 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
478}
479
480func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
481 bp := cc.GatherRequiredDepsForTest(android.Android) + `
482 android_app {
483 name: "foo",
484 srcs: ["a.java"],
485 updatable: true,
486 sdk_version: "current",
487 min_sdk_version: "29", // this APK should support 29
488 jni_libs: ["libjni"],
489 }
490
491 cc_library {
492 name: "libjni",
493 stl: "none",
494 shared_libs: ["libbar"],
495 system_shared_libs: [],
496 sdk_version: "27",
497 }
498
499 cc_library {
500 name: "libbar",
501 stl: "none",
502 system_shared_libs: [],
503 sdk_version: "current",
504 }
505 `
506 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
507}
508
Colin Cross0ddae7f2019-02-07 15:30:01 -0800509func TestResourceDirs(t *testing.T) {
510 testCases := []struct {
511 name string
512 prop string
513 resources []string
514 }{
515 {
516 name: "no resource_dirs",
517 prop: "",
518 resources: []string{"res/res/values/strings.xml"},
519 },
520 {
521 name: "resource_dirs",
522 prop: `resource_dirs: ["res"]`,
523 resources: []string{"res/res/values/strings.xml"},
524 },
525 {
526 name: "empty resource_dirs",
527 prop: `resource_dirs: []`,
528 resources: nil,
529 },
530 }
531
532 fs := map[string][]byte{
533 "res/res/values/strings.xml": nil,
534 }
535
536 bp := `
537 android_app {
538 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900539 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800540 %s
541 }
542 `
543
544 for _, testCase := range testCases {
545 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800546 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
547 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800548 run(t, ctx, config)
549
550 module := ctx.ModuleForTests("foo", "android_common")
551 resourceList := module.MaybeOutput("aapt2/res.list")
552
553 var resources []string
554 if resourceList.Rule != nil {
555 for _, compiledResource := range resourceList.Inputs.Strings() {
556 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
557 }
558 }
559
560 if !reflect.DeepEqual(resources, testCase.resources) {
561 t.Errorf("expected resource files %q, got %q",
562 testCase.resources, resources)
563 }
564 })
565 }
566}
567
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800568func TestLibraryAssets(t *testing.T) {
569 bp := `
570 android_app {
571 name: "foo",
572 sdk_version: "current",
573 static_libs: ["lib1", "lib2", "lib3"],
574 }
575
576 android_library {
577 name: "lib1",
578 sdk_version: "current",
579 asset_dirs: ["assets_a"],
580 }
581
582 android_library {
583 name: "lib2",
584 sdk_version: "current",
585 }
586
587 android_library {
588 name: "lib3",
589 sdk_version: "current",
590 static_libs: ["lib4"],
591 }
592
593 android_library {
594 name: "lib4",
595 sdk_version: "current",
596 asset_dirs: ["assets_b"],
597 }
598 `
599
600 testCases := []struct {
601 name string
602 assetFlag string
603 assetPackages []string
604 }{
605 {
606 name: "foo",
607 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
608 assetPackages: []string{
609 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
610 buildDir + "/.intermediates/lib1/android_common/assets.zip",
611 buildDir + "/.intermediates/lib3/android_common/assets.zip",
612 },
613 },
614 {
615 name: "lib1",
616 assetFlag: "-A assets_a",
617 },
618 {
619 name: "lib2",
620 },
621 {
622 name: "lib3",
623 assetPackages: []string{
624 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
625 buildDir + "/.intermediates/lib4/android_common/assets.zip",
626 },
627 },
628 {
629 name: "lib4",
630 assetFlag: "-A assets_b",
631 },
632 }
633 ctx := testApp(t, bp)
634
635 for _, test := range testCases {
636 t.Run(test.name, func(t *testing.T) {
637 m := ctx.ModuleForTests(test.name, "android_common")
638
639 // Check asset flag in aapt2 link flags
640 var aapt2link android.TestingBuildParams
641 if len(test.assetPackages) > 0 {
642 aapt2link = m.Output("aapt2/package-res.apk")
643 } else {
644 aapt2link = m.Output("package-res.apk")
645 }
646 aapt2Flags := aapt2link.Args["flags"]
647 if test.assetFlag != "" {
648 if !strings.Contains(aapt2Flags, test.assetFlag) {
649 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
650 }
651 } else {
652 if strings.Contains(aapt2Flags, " -A ") {
653 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
654 }
655 }
656
657 // Check asset merge rule.
658 if len(test.assetPackages) > 0 {
659 mergeAssets := m.Output("package-res.apk")
660 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
661 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
662 mergeAssets.Inputs.Strings(), test.assetPackages)
663 }
664 }
665 })
666 }
667}
668
Colin Crossbec85302019-02-13 13:15:46 -0800669func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800670 testCases := []struct {
671 name string
672 enforceRROTargets []string
673 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800674 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800675 overlayFiles map[string][]string
676 rroDirs map[string][]string
677 }{
678 {
679 name: "no RRO",
680 enforceRROTargets: nil,
681 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800682 resourceFiles: map[string][]string{
683 "foo": nil,
684 "bar": {"bar/res/res/values/strings.xml"},
685 "lib": nil,
686 "lib2": {"lib2/res/res/values/strings.xml"},
687 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800688 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800689 "foo": {
690 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800691 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000692 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800693 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800694 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
695 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000696 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800697 },
Colin Crossbec85302019-02-13 13:15:46 -0800698 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800699 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
700 "device/vendor/blah/overlay/bar/res/values/strings.xml",
701 },
Colin Crossbec85302019-02-13 13:15:46 -0800702 "lib": {
703 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
704 "lib/res/res/values/strings.xml",
705 "device/vendor/blah/overlay/lib/res/values/strings.xml",
706 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800707 },
708 rroDirs: map[string][]string{
709 "foo": nil,
710 "bar": nil,
711 },
712 },
713 {
714 name: "enforce RRO on foo",
715 enforceRROTargets: []string{"foo"},
716 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800717 resourceFiles: map[string][]string{
718 "foo": nil,
719 "bar": {"bar/res/res/values/strings.xml"},
720 "lib": nil,
721 "lib2": {"lib2/res/res/values/strings.xml"},
722 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800723 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800724 "foo": {
725 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800726 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000727 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800728 "foo/res/res/values/strings.xml",
729 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
730 },
Colin Crossbec85302019-02-13 13:15:46 -0800731 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800732 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
733 "device/vendor/blah/overlay/bar/res/values/strings.xml",
734 },
Colin Crossbec85302019-02-13 13:15:46 -0800735 "lib": {
736 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
737 "lib/res/res/values/strings.xml",
738 "device/vendor/blah/overlay/lib/res/values/strings.xml",
739 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800740 },
Colin Crossc1c37552019-01-31 11:42:41 -0800741
Colin Cross5c4791c2019-02-01 11:44:44 -0800742 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800743 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000744 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800745 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
746 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000747 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800748 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800749 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800750 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800751 },
752 },
753 {
754 name: "enforce RRO on all",
755 enforceRROTargets: []string{"*"},
756 enforceRROExcludedOverlays: []string{
757 // Excluding specific apps/res directories also allowed.
758 "device/vendor/blah/static_overlay/foo",
759 "device/vendor/blah/static_overlay/bar/res",
760 },
Colin Crossbec85302019-02-13 13:15:46 -0800761 resourceFiles: map[string][]string{
762 "foo": nil,
763 "bar": {"bar/res/res/values/strings.xml"},
764 "lib": nil,
765 "lib2": {"lib2/res/res/values/strings.xml"},
766 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800767 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800768 "foo": {
769 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800770 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000771 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800772 "foo/res/res/values/strings.xml",
773 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
774 },
Colin Crossbec85302019-02-13 13:15:46 -0800775 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
776 "lib": {
777 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
778 "lib/res/res/values/strings.xml",
779 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800780 },
781 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800782 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000783 "device:device/vendor/blah/overlay/foo/res",
784 "product:product/vendor/blah/overlay/foo/res",
785 // Lib dep comes after the direct deps
786 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800787 },
Anton Hansson53c88442019-03-18 15:53:16 +0000788 "bar": {"device:device/vendor/blah/overlay/bar/res"},
789 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800790 },
791 },
792 }
793
Anton Hansson53c88442019-03-18 15:53:16 +0000794 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800795 "device/vendor/blah/overlay",
796 "device/vendor/blah/overlay2",
797 "device/vendor/blah/static_overlay",
798 }
799
Anton Hansson53c88442019-03-18 15:53:16 +0000800 productResourceOverlays := []string{
801 "product/vendor/blah/overlay",
802 }
803
Colin Cross890ff552017-11-30 20:13:19 -0800804 fs := map[string][]byte{
805 "foo/res/res/values/strings.xml": nil,
806 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800807 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800808 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800809 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
810 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800811 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800812 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
813 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
814 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000815 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800816 }
817
818 bp := `
819 android_app {
820 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900821 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800822 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000823 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800824 }
825
826 android_app {
827 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900828 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800829 resource_dirs: ["bar/res"],
830 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800831
832 android_library {
833 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900834 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800835 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800836 static_libs: ["lib2"],
837 }
838
839 android_library {
840 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900841 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800842 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800843 }
Anton Hansson53c88442019-03-18 15:53:16 +0000844
845 // This library has the same resources as lib (should not lead to dupe RROs)
846 android_library {
847 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900848 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000849 resource_dirs: ["lib/res"]
850 }
Colin Cross890ff552017-11-30 20:13:19 -0800851 `
852
Colin Cross5c4791c2019-02-01 11:44:44 -0800853 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800854 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800855 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000856 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
857 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800858 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800859 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800860 }
861 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800862 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800863 }
864
Colin Cross98be1bb2019-12-13 20:41:13 -0800865 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800866 run(t, ctx, config)
867
Colin Crossbec85302019-02-13 13:15:46 -0800868 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
869 for _, o := range list {
870 res := module.MaybeOutput(o)
871 if res.Rule != nil {
872 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
873 // verify the inputs to the .arsc.flat rule.
874 files = append(files, res.Inputs.Strings()...)
875 } else {
876 // Otherwise, verify the full path to the output of the other module
877 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000878 }
Colin Cross890ff552017-11-30 20:13:19 -0800879 }
Colin Crossbec85302019-02-13 13:15:46 -0800880 return files
Colin Cross890ff552017-11-30 20:13:19 -0800881 }
882
Colin Crossbec85302019-02-13 13:15:46 -0800883 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
884 module := ctx.ModuleForTests(moduleName, "android_common")
885 resourceList := module.MaybeOutput("aapt2/res.list")
886 if resourceList.Rule != nil {
887 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000888 }
Colin Crossbec85302019-02-13 13:15:46 -0800889 overlayList := module.MaybeOutput("aapt2/overlay.list")
890 if overlayList.Rule != nil {
891 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
892 }
893
Anton Hansson53c88442019-03-18 15:53:16 +0000894 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
895 var prefix string
896 if d.overlayType == device {
897 prefix = "device:"
898 } else if d.overlayType == product {
899 prefix = "product:"
900 } else {
901 t.Fatalf("Unexpected overlayType %d", d.overlayType)
902 }
903 rroDirs = append(rroDirs, prefix+d.path.String())
904 }
Colin Crossbec85302019-02-13 13:15:46 -0800905
906 return resourceFiles, overlayFiles, rroDirs
907 }
908
909 modules := []string{"foo", "bar", "lib", "lib2"}
910 for _, module := range modules {
911 resourceFiles, overlayFiles, rroDirs := getResources(module)
912
913 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
914 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
915 module, testCase.resourceFiles[module], resourceFiles)
916 }
917 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
918 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
919 module, testCase.overlayFiles[module], overlayFiles)
920 }
921 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +0000922 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -0800923 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +0000924 }
Colin Cross890ff552017-11-30 20:13:19 -0800925 }
Colin Cross890ff552017-11-30 20:13:19 -0800926 })
927 }
928}
Colin Crossd09b0b62018-04-18 11:06:47 -0700929
930func TestAppSdkVersion(t *testing.T) {
931 testCases := []struct {
932 name string
933 sdkVersion string
934 platformSdkInt int
935 platformSdkCodename string
936 platformSdkFinal bool
937 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +0900938 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -0700939 }{
940 {
941 name: "current final SDK",
942 sdkVersion: "current",
943 platformSdkInt: 27,
944 platformSdkCodename: "REL",
945 platformSdkFinal: true,
946 expectedMinSdkVersion: "27",
947 },
948 {
949 name: "current non-final SDK",
950 sdkVersion: "current",
951 platformSdkInt: 27,
952 platformSdkCodename: "OMR1",
953 platformSdkFinal: false,
954 expectedMinSdkVersion: "OMR1",
955 },
956 {
957 name: "default final SDK",
958 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900959 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700960 platformSdkInt: 27,
961 platformSdkCodename: "REL",
962 platformSdkFinal: true,
963 expectedMinSdkVersion: "27",
964 },
965 {
966 name: "default non-final SDK",
967 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900968 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700969 platformSdkInt: 27,
970 platformSdkCodename: "OMR1",
971 platformSdkFinal: false,
972 expectedMinSdkVersion: "OMR1",
973 },
974 {
975 name: "14",
976 sdkVersion: "14",
977 expectedMinSdkVersion: "14",
978 },
979 }
980
981 for _, moduleType := range []string{"android_app", "android_library"} {
982 for _, test := range testCases {
983 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +0900984 platformApiProp := ""
985 if test.platformApis {
986 platformApiProp = "platform_apis: true,"
987 }
Colin Crossd09b0b62018-04-18 11:06:47 -0700988 bp := fmt.Sprintf(`%s {
989 name: "foo",
990 srcs: ["a.java"],
991 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900992 %s
993 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -0700994
Colin Cross98be1bb2019-12-13 20:41:13 -0800995 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -0700996 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
997 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
998 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
999
Colin Cross98be1bb2019-12-13 20:41:13 -08001000 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -07001001
1002 run(t, ctx, config)
1003
1004 foo := ctx.ModuleForTests("foo", "android_common")
1005 link := foo.Output("package-res.apk")
1006 linkFlags := strings.Split(link.Args["flags"], " ")
1007 min := android.IndexList("--min-sdk-version", linkFlags)
1008 target := android.IndexList("--target-sdk-version", linkFlags)
1009
1010 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1011 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1012 }
1013
1014 gotMinSdkVersion := linkFlags[min+1]
1015 gotTargetSdkVersion := linkFlags[target+1]
1016
1017 if gotMinSdkVersion != test.expectedMinSdkVersion {
1018 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1019 test.expectedMinSdkVersion, gotMinSdkVersion)
1020 }
1021
1022 if gotTargetSdkVersion != test.expectedMinSdkVersion {
1023 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1024 test.expectedMinSdkVersion, gotTargetSdkVersion)
1025 }
1026 })
1027 }
1028 }
1029}
Colin Crossa4f08812018-10-02 22:03:40 -07001030
Paul Duffin50c217c2019-06-12 13:25:22 +01001031func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001032 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001033 cc_library {
1034 name: "libjni",
1035 system_shared_libs: [],
Colin Crossc511bc52020-04-07 16:50:32 +00001036 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001037 stl: "none",
1038 }
1039
1040 android_test {
1041 name: "test",
1042 sdk_version: "core_platform",
1043 jni_libs: ["libjni"],
1044 }
1045
1046 android_test {
1047 name: "test_first",
1048 sdk_version: "core_platform",
1049 compile_multilib: "first",
1050 jni_libs: ["libjni"],
1051 }
1052
1053 android_test {
1054 name: "test_both",
1055 sdk_version: "core_platform",
1056 compile_multilib: "both",
1057 jni_libs: ["libjni"],
1058 }
1059
1060 android_test {
1061 name: "test_32",
1062 sdk_version: "core_platform",
1063 compile_multilib: "32",
1064 jni_libs: ["libjni"],
1065 }
1066
1067 android_test {
1068 name: "test_64",
1069 sdk_version: "core_platform",
1070 compile_multilib: "64",
1071 jni_libs: ["libjni"],
1072 }
1073 `)
1074
1075 testCases := []struct {
1076 name string
1077 abis []string
1078 }{
1079 {"test", []string{"arm64-v8a"}},
1080 {"test_first", []string{"arm64-v8a"}},
1081 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1082 {"test_32", []string{"armeabi-v7a"}},
1083 {"test_64", []string{"arm64-v8a"}},
1084 }
1085
1086 for _, test := range testCases {
1087 t.Run(test.name, func(t *testing.T) {
1088 app := ctx.ModuleForTests(test.name, "android_common")
1089 jniLibZip := app.Output("jnilibs.zip")
1090 var abis []string
1091 args := strings.Fields(jniLibZip.Args["jarArgs"])
1092 for i := 0; i < len(args); i++ {
1093 if args[i] == "-P" {
1094 abis = append(abis, filepath.Base(args[i+1]))
1095 i++
1096 }
1097 }
1098 if !reflect.DeepEqual(abis, test.abis) {
1099 t.Errorf("want abis %v, got %v", test.abis, abis)
1100 }
1101 })
1102 }
1103}
1104
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001105func TestAppSdkVersionByPartition(t *testing.T) {
1106 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1107 android_app {
1108 name: "foo",
1109 srcs: ["a.java"],
1110 vendor: true,
1111 platform_apis: true,
1112 }
1113 `)
1114
1115 testJava(t, `
1116 android_app {
1117 name: "bar",
1118 srcs: ["b.java"],
1119 platform_apis: true,
1120 }
1121 `)
1122
1123 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001124 bp := `
1125 android_app {
1126 name: "foo",
1127 srcs: ["a.java"],
1128 product_specific: true,
1129 platform_apis: true,
1130 }
1131 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001132
1133 config := testAppConfig(nil, bp, nil)
1134 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001135 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001136 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 +09001137 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001138 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001139 }
1140 }
1141}
1142
Paul Duffin50c217c2019-06-12 13:25:22 +01001143func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001144 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001145 cc_library {
1146 name: "libjni",
1147 system_shared_libs: [],
1148 stl: "none",
Colin Cross094cde42020-02-15 10:38:00 -08001149 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001150 }
1151
1152 android_app {
1153 name: "app",
1154 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001155 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001156 }
1157
1158 android_app {
1159 name: "app_noembed",
1160 jni_libs: ["libjni"],
1161 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001162 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001163 }
1164
1165 android_app {
1166 name: "app_embed",
1167 jni_libs: ["libjni"],
1168 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001169 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001170 }
1171
1172 android_test {
1173 name: "test",
Colin Crossc511bc52020-04-07 16:50:32 +00001174 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001175 jni_libs: ["libjni"],
1176 }
1177
1178 android_test {
1179 name: "test_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001180 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001181 jni_libs: ["libjni"],
1182 use_embedded_native_libs: false,
1183 }
1184
1185 android_test_helper_app {
1186 name: "test_helper",
Colin Crossc511bc52020-04-07 16:50:32 +00001187 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001188 jni_libs: ["libjni"],
1189 }
1190
1191 android_test_helper_app {
1192 name: "test_helper_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001193 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001194 jni_libs: ["libjni"],
1195 use_embedded_native_libs: false,
1196 }
1197 `)
1198
1199 testCases := []struct {
1200 name string
1201 packaged bool
1202 compressed bool
1203 }{
1204 {"app", false, false},
1205 {"app_noembed", false, false},
1206 {"app_embed", true, false},
1207 {"test", true, false},
1208 {"test_noembed", true, true},
1209 {"test_helper", true, false},
1210 {"test_helper_noembed", true, true},
1211 }
1212
1213 for _, test := range testCases {
1214 t.Run(test.name, func(t *testing.T) {
1215 app := ctx.ModuleForTests(test.name, "android_common")
1216 jniLibZip := app.MaybeOutput("jnilibs.zip")
1217 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1218 t.Errorf("expected jni packaged %v, got %v", w, g)
1219 }
1220
1221 if jniLibZip.Rule != nil {
1222 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1223 t.Errorf("expected jni compressed %v, got %v", w, g)
1224 }
Colin Crossc511bc52020-04-07 16:50:32 +00001225
1226 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1227 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1228 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001229 }
1230 })
1231 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001232}
1233
Colin Cross3c007702020-05-08 11:20:24 -07001234func TestJNISDK(t *testing.T) {
1235 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1236 cc_library {
1237 name: "libjni",
1238 system_shared_libs: [],
1239 stl: "none",
1240 sdk_version: "current",
1241 }
1242
1243 android_test {
1244 name: "app_platform",
1245 jni_libs: ["libjni"],
1246 platform_apis: true,
1247 }
1248
1249 android_test {
1250 name: "app_sdk",
1251 jni_libs: ["libjni"],
1252 sdk_version: "current",
1253 }
1254
1255 android_test {
1256 name: "app_force_platform",
1257 jni_libs: ["libjni"],
1258 sdk_version: "current",
1259 jni_uses_platform_apis: true,
1260 }
1261
1262 android_test {
1263 name: "app_force_sdk",
1264 jni_libs: ["libjni"],
1265 platform_apis: true,
1266 jni_uses_sdk_apis: true,
1267 }
Colin Crossc2d24052020-05-13 11:05:02 -07001268
1269 cc_library {
1270 name: "libvendorjni",
1271 system_shared_libs: [],
1272 stl: "none",
1273 vendor: true,
1274 }
1275
1276 android_test {
1277 name: "app_vendor",
1278 jni_libs: ["libvendorjni"],
1279 sdk_version: "current",
1280 vendor: true,
1281 }
Colin Cross3c007702020-05-08 11:20:24 -07001282 `)
1283
1284 testCases := []struct {
Colin Crossc2d24052020-05-13 11:05:02 -07001285 name string
1286 sdkJNI bool
1287 vendorJNI bool
Colin Cross3c007702020-05-08 11:20:24 -07001288 }{
Colin Crossc2d24052020-05-13 11:05:02 -07001289 {name: "app_platform"},
1290 {name: "app_sdk", sdkJNI: true},
1291 {name: "app_force_platform"},
1292 {name: "app_force_sdk", sdkJNI: true},
1293 {name: "app_vendor", vendorJNI: true},
Colin Cross3c007702020-05-08 11:20:24 -07001294 }
1295
Colin Crossc2d24052020-05-13 11:05:02 -07001296 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1297 Output("libjni.so").Output.String()
1298 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1299 Output("libjni.so").Output.String()
1300 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1301 Output("libvendorjni.so").Output.String()
1302
Colin Cross3c007702020-05-08 11:20:24 -07001303 for _, test := range testCases {
1304 t.Run(test.name, func(t *testing.T) {
1305 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross3c007702020-05-08 11:20:24 -07001306
1307 jniLibZip := app.MaybeOutput("jnilibs.zip")
1308 if len(jniLibZip.Implicits) != 1 {
1309 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1310 }
1311 gotJNI := jniLibZip.Implicits[0].String()
1312
1313 if test.sdkJNI {
1314 if gotJNI != sdkJNI {
1315 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1316 }
Colin Crossc2d24052020-05-13 11:05:02 -07001317 } else if test.vendorJNI {
1318 if gotJNI != vendorJNI {
1319 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1320 }
Colin Cross3c007702020-05-08 11:20:24 -07001321 } else {
1322 if gotJNI != platformJNI {
1323 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1324 }
1325 }
1326 })
1327 }
1328
1329 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1330 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1331 android_test {
1332 name: "app_platform",
1333 platform_apis: true,
1334 jni_uses_platform_apis: true,
1335 }
1336 `)
1337 })
1338
1339 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1340 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1341 android_test {
1342 name: "app_sdk",
1343 sdk_version: "current",
1344 jni_uses_sdk_apis: true,
1345 }
1346 `)
1347 })
1348
1349}
1350
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001351func TestCertificates(t *testing.T) {
1352 testCases := []struct {
1353 name string
1354 bp string
1355 certificateOverride string
Liz Kammere2b27f42020-05-07 13:24:05 -07001356 expectedLineage string
1357 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001358 }{
1359 {
1360 name: "default",
1361 bp: `
1362 android_app {
1363 name: "foo",
1364 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001365 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001366 }
1367 `,
1368 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001369 expectedLineage: "",
1370 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001371 },
1372 {
1373 name: "module certificate property",
1374 bp: `
1375 android_app {
1376 name: "foo",
1377 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001378 certificate: ":new_certificate",
1379 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001380 }
1381
1382 android_app_certificate {
1383 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001384 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001385 }
1386 `,
1387 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001388 expectedLineage: "",
1389 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001390 },
1391 {
1392 name: "path certificate property",
1393 bp: `
1394 android_app {
1395 name: "foo",
1396 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001397 certificate: "expiredkey",
1398 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001399 }
1400 `,
1401 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001402 expectedLineage: "",
1403 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001404 },
1405 {
1406 name: "certificate overrides",
1407 bp: `
1408 android_app {
1409 name: "foo",
1410 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001411 certificate: "expiredkey",
1412 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001413 }
1414
1415 android_app_certificate {
1416 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001417 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001418 }
1419 `,
1420 certificateOverride: "foo:new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001421 expectedLineage: "",
1422 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1423 },
1424 {
1425 name: "certificate lineage",
1426 bp: `
1427 android_app {
1428 name: "foo",
1429 srcs: ["a.java"],
1430 certificate: ":new_certificate",
1431 lineage: "lineage.bin",
1432 sdk_version: "current",
1433 }
1434
1435 android_app_certificate {
1436 name: "new_certificate",
1437 certificate: "cert/new_cert",
1438 }
1439 `,
1440 certificateOverride: "",
1441 expectedLineage: "--lineage lineage.bin",
1442 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001443 },
1444 }
1445
1446 for _, test := range testCases {
1447 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001448 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001449 if test.certificateOverride != "" {
1450 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1451 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001452 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001453
1454 run(t, ctx, config)
1455 foo := ctx.ModuleForTests("foo", "android_common")
1456
1457 signapk := foo.Output("foo.apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001458 signCertificateFlags := signapk.Args["certificates"]
1459 if test.expectedCertificate != signCertificateFlags {
1460 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1461 }
1462
1463 signFlags := signapk.Args["flags"]
1464 if test.expectedLineage != signFlags {
1465 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001466 }
1467 })
1468 }
1469}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001470
1471func TestPackageNameOverride(t *testing.T) {
1472 testCases := []struct {
1473 name string
1474 bp string
1475 packageNameOverride string
1476 expected []string
1477 }{
1478 {
1479 name: "default",
1480 bp: `
1481 android_app {
1482 name: "foo",
1483 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001484 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001485 }
1486 `,
1487 packageNameOverride: "",
1488 expected: []string{
1489 buildDir + "/.intermediates/foo/android_common/foo.apk",
1490 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1491 },
1492 },
1493 {
1494 name: "overridden",
1495 bp: `
1496 android_app {
1497 name: "foo",
1498 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001499 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001500 }
1501 `,
1502 packageNameOverride: "foo:bar",
1503 expected: []string{
1504 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001505 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001506 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1507 },
1508 },
1509 }
1510
1511 for _, test := range testCases {
1512 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001513 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001514 if test.packageNameOverride != "" {
1515 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1516 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001517 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001518
1519 run(t, ctx, config)
1520 foo := ctx.ModuleForTests("foo", "android_common")
1521
1522 outputs := foo.AllOutputs()
1523 outputMap := make(map[string]bool)
1524 for _, o := range outputs {
1525 outputMap[o] = true
1526 }
1527 for _, e := range test.expected {
1528 if _, exist := outputMap[e]; !exist {
1529 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1530 }
1531 }
1532 })
1533 }
1534}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001535
1536func TestInstrumentationTargetOverridden(t *testing.T) {
1537 bp := `
1538 android_app {
1539 name: "foo",
1540 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001541 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001542 }
1543
1544 android_test {
1545 name: "bar",
1546 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001547 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001548 }
1549 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001550 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001551 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001552 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001553
1554 run(t, ctx, config)
1555
1556 bar := ctx.ModuleForTests("bar", "android_common")
1557 res := bar.Output("package-res.apk")
1558 aapt2Flags := res.Args["flags"]
1559 e := "--rename-instrumentation-target-package org.dandroid.bp"
1560 if !strings.Contains(aapt2Flags, e) {
1561 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1562 }
1563}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001564
1565func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001566 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001567 android_app {
1568 name: "foo",
1569 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001570 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001571 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001572 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001573 }
1574
1575 override_android_app {
1576 name: "bar",
1577 base: "foo",
1578 certificate: ":new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001579 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001580 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001581 }
1582
1583 android_app_certificate {
1584 name: "new_certificate",
1585 certificate: "cert/new_cert",
1586 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001587
1588 override_android_app {
1589 name: "baz",
1590 base: "foo",
1591 package_name: "org.dandroid.bp",
1592 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001593 `)
1594
1595 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001596 moduleName string
1597 variantName string
1598 apkName string
1599 apkPath string
Liz Kammere2b27f42020-05-07 13:24:05 -07001600 certFlag string
1601 lineageFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001602 overrides []string
1603 aaptFlag string
1604 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001605 }{
1606 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001607 moduleName: "foo",
1608 variantName: "android_common",
1609 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001610 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1611 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001612 overrides: []string{"qux"},
1613 aaptFlag: "",
1614 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001615 },
1616 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001617 moduleName: "bar",
1618 variantName: "android_common_bar",
1619 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001620 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1621 lineageFlag: "--lineage lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001622 overrides: []string{"qux", "foo"},
1623 aaptFlag: "",
1624 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001625 },
1626 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001627 moduleName: "baz",
1628 variantName: "android_common_baz",
1629 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001630 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1631 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001632 overrides: []string{"qux", "foo"},
1633 aaptFlag: "--rename-manifest-package org.dandroid.bp",
1634 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001635 },
1636 }
1637 for _, expected := range expectedVariants {
1638 variant := ctx.ModuleForTests("foo", expected.variantName)
1639
1640 // Check the final apk name
1641 outputs := variant.AllOutputs()
1642 expectedApkPath := buildDir + expected.apkPath
1643 found := false
1644 for _, o := range outputs {
1645 if o == expectedApkPath {
1646 found = true
1647 break
1648 }
1649 }
1650 if !found {
1651 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1652 }
1653
1654 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001655 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001656 certFlag := signapk.Args["certificates"]
1657 if expected.certFlag != certFlag {
1658 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1659 }
1660
1661 // Check the lineage flags
1662 lineageFlag := signapk.Args["flags"]
1663 if expected.lineageFlag != lineageFlag {
1664 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001665 }
1666
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001667 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001668 mod := variant.Module().(*AndroidApp)
1669 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1670 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1671 expected.overrides, mod.appProperties.Overrides)
1672 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001673
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001674 // Test Overridable property: Logging_parent
1675 logging_parent := mod.aapt.LoggingParent
1676 if expected.logging_parent != logging_parent {
1677 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1678 expected.logging_parent, logging_parent)
1679 }
1680
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001681 // Check the package renaming flag, if exists.
1682 res := variant.Output("package-res.apk")
1683 aapt2Flags := res.Args["flags"]
1684 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1685 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1686 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001687 }
1688}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001689
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001690func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001691 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001692 android_app {
1693 name: "foo",
1694 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001695 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001696 }
1697
1698 override_android_app {
1699 name: "bar",
1700 base: "foo",
1701 package_name: "org.dandroid.bp",
1702 }
1703
1704 android_test {
1705 name: "baz",
1706 srcs: ["b.java"],
1707 instrumentation_for: "foo",
1708 }
1709
1710 android_test {
1711 name: "qux",
1712 srcs: ["b.java"],
1713 instrumentation_for: "bar",
1714 }
1715 `)
1716
1717 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1718 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1719 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1720 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1721 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1722 }
1723
1724 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1725 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1726 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1727 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1728 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1729 }
1730}
1731
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001732func TestOverrideAndroidTest(t *testing.T) {
1733 ctx, _ := testJava(t, `
1734 android_app {
1735 name: "foo",
1736 srcs: ["a.java"],
1737 package_name: "com.android.foo",
1738 sdk_version: "current",
1739 }
1740
1741 override_android_app {
1742 name: "bar",
1743 base: "foo",
1744 package_name: "com.android.bar",
1745 }
1746
1747 android_test {
1748 name: "foo_test",
1749 srcs: ["b.java"],
1750 instrumentation_for: "foo",
1751 }
1752
1753 override_android_test {
1754 name: "bar_test",
1755 base: "foo_test",
1756 package_name: "com.android.bar.test",
1757 instrumentation_for: "bar",
1758 instrumentation_target_package: "com.android.bar",
1759 }
1760 `)
1761
1762 expectedVariants := []struct {
1763 moduleName string
1764 variantName string
1765 apkPath string
1766 overrides []string
1767 targetVariant string
1768 packageFlag string
1769 targetPackageFlag string
1770 }{
1771 {
1772 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001773 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001774 overrides: nil,
1775 targetVariant: "android_common",
1776 packageFlag: "",
1777 targetPackageFlag: "",
1778 },
1779 {
1780 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001781 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001782 overrides: []string{"foo_test"},
1783 targetVariant: "android_common_bar",
1784 packageFlag: "com.android.bar.test",
1785 targetPackageFlag: "com.android.bar",
1786 },
1787 }
1788 for _, expected := range expectedVariants {
1789 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1790
1791 // Check the final apk name
1792 outputs := variant.AllOutputs()
1793 expectedApkPath := buildDir + expected.apkPath
1794 found := false
1795 for _, o := range outputs {
1796 if o == expectedApkPath {
1797 found = true
1798 break
1799 }
1800 }
1801 if !found {
1802 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1803 }
1804
1805 // Check if the overrides field values are correctly aggregated.
1806 mod := variant.Module().(*AndroidTest)
1807 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1808 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1809 expected.overrides, mod.appProperties.Overrides)
1810 }
1811
1812 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1813 javac := variant.Rule("javac")
1814 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1815 if !strings.Contains(javac.Args["classpath"], turbine) {
1816 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1817 }
1818
1819 // Check aapt2 flags.
1820 res := variant.Output("package-res.apk")
1821 aapt2Flags := res.Args["flags"]
1822 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1823 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1824 }
1825}
1826
Jaewoong Jung39982342020-01-14 10:27:18 -08001827func TestAndroidTest_FixTestConfig(t *testing.T) {
1828 ctx, _ := testJava(t, `
1829 android_app {
1830 name: "foo",
1831 srcs: ["a.java"],
1832 package_name: "com.android.foo",
1833 sdk_version: "current",
1834 }
1835
1836 android_test {
1837 name: "foo_test",
1838 srcs: ["b.java"],
1839 instrumentation_for: "foo",
1840 }
1841
1842 android_test {
1843 name: "bar_test",
1844 srcs: ["b.java"],
1845 package_name: "com.android.bar.test",
1846 instrumentation_for: "foo",
1847 }
1848
1849 override_android_test {
1850 name: "baz_test",
1851 base: "foo_test",
1852 package_name: "com.android.baz.test",
1853 }
1854 `)
1855
1856 testCases := []struct {
1857 moduleName string
1858 variantName string
1859 expectedFlags []string
1860 }{
1861 {
1862 moduleName: "foo_test",
1863 variantName: "android_common",
1864 },
1865 {
1866 moduleName: "bar_test",
1867 variantName: "android_common",
1868 expectedFlags: []string{
1869 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
1870 "--package-name com.android.bar.test",
1871 },
1872 },
1873 {
1874 moduleName: "foo_test",
1875 variantName: "android_common_baz_test",
1876 expectedFlags: []string{
1877 "--manifest " + buildDir +
1878 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
1879 "--package-name com.android.baz.test",
1880 "--test-file-name baz_test.apk",
1881 },
1882 },
1883 }
1884
1885 for _, test := range testCases {
1886 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
1887 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
1888
1889 if len(test.expectedFlags) > 0 {
1890 if params.Rule == nil {
1891 t.Errorf("test_config_fixer was expected to run, but didn't")
1892 } else {
1893 for _, flag := range test.expectedFlags {
1894 if !strings.Contains(params.RuleParams.Command, flag) {
1895 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
1896 }
1897 }
1898 }
1899 } else {
1900 if params.Rule != nil {
1901 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
1902 }
1903 }
1904
1905 }
1906}
1907
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001908func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001909 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001910 android_app_import {
1911 name: "foo",
1912 apk: "prebuilts/apk/app.apk",
1913 certificate: "platform",
1914 dex_preopt: {
1915 enabled: true,
1916 },
1917 }
1918 `)
1919
1920 variant := ctx.ModuleForTests("foo", "android_common")
1921
1922 // Check dexpreopt outputs.
1923 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1924 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1925 t.Errorf("can't find dexpreopt outputs")
1926 }
1927
1928 // Check cert signing flag.
1929 signedApk := variant.Output("signed/foo.apk")
1930 signingFlag := signedApk.Args["certificates"]
1931 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
1932 if expected != signingFlag {
1933 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1934 }
1935}
1936
1937func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001938 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001939 android_app_import {
1940 name: "foo",
1941 apk: "prebuilts/apk/app.apk",
1942 certificate: "platform",
1943 dex_preopt: {
1944 enabled: false,
1945 },
1946 }
1947 `)
1948
1949 variant := ctx.ModuleForTests("foo", "android_common")
1950
1951 // Check dexpreopt outputs. They shouldn't exist.
1952 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
1953 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
1954 t.Errorf("dexpreopt shouldn't have run.")
1955 }
1956}
1957
1958func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001959 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001960 android_app_import {
1961 name: "foo",
1962 apk: "prebuilts/apk/app.apk",
1963 presigned: true,
1964 dex_preopt: {
1965 enabled: true,
1966 },
1967 }
1968 `)
1969
1970 variant := ctx.ModuleForTests("foo", "android_common")
1971
1972 // Check dexpreopt outputs.
1973 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1974 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1975 t.Errorf("can't find dexpreopt outputs")
1976 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01001977 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001978 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
1979 t.Errorf("signing rule shouldn't be included.")
1980 }
1981 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
1982 t.Errorf("can't find aligning rule")
1983 }
1984}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001985
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001986func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
1987 ctx, _ := testJava(t, `
1988 android_app_import {
1989 name: "foo",
1990 apk: "prebuilts/apk/app.apk",
1991 default_dev_cert: true,
1992 dex_preopt: {
1993 enabled: true,
1994 },
1995 }
1996 `)
1997
1998 variant := ctx.ModuleForTests("foo", "android_common")
1999
2000 // Check dexpreopt outputs.
2001 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2002 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2003 t.Errorf("can't find dexpreopt outputs")
2004 }
2005
2006 // Check cert signing flag.
2007 signedApk := variant.Output("signed/foo.apk")
2008 signingFlag := signedApk.Args["certificates"]
2009 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2010 if expected != signingFlag {
2011 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2012 }
2013}
2014
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002015func TestAndroidAppImport_DpiVariants(t *testing.T) {
2016 bp := `
2017 android_app_import {
2018 name: "foo",
2019 apk: "prebuilts/apk/app.apk",
2020 dpi_variants: {
2021 xhdpi: {
2022 apk: "prebuilts/apk/app_xhdpi.apk",
2023 },
2024 xxhdpi: {
2025 apk: "prebuilts/apk/app_xxhdpi.apk",
2026 },
2027 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002028 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002029 dex_preopt: {
2030 enabled: true,
2031 },
2032 }
2033 `
2034 testCases := []struct {
2035 name string
2036 aaptPreferredConfig *string
2037 aaptPrebuiltDPI []string
2038 expected string
2039 }{
2040 {
2041 name: "no preferred",
2042 aaptPreferredConfig: nil,
2043 aaptPrebuiltDPI: []string{},
2044 expected: "prebuilts/apk/app.apk",
2045 },
2046 {
2047 name: "AAPTPreferredConfig matches",
2048 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002049 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002050 expected: "prebuilts/apk/app_xhdpi.apk",
2051 },
2052 {
2053 name: "AAPTPrebuiltDPI matches",
2054 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2055 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2056 expected: "prebuilts/apk/app_xxhdpi.apk",
2057 },
2058 {
2059 name: "non-first AAPTPrebuiltDPI matches",
2060 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2061 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2062 expected: "prebuilts/apk/app_xhdpi.apk",
2063 },
2064 {
2065 name: "no matches",
2066 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2067 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2068 expected: "prebuilts/apk/app.apk",
2069 },
2070 }
2071
2072 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2073 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002074 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002075 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2076 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002077 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002078
2079 run(t, ctx, config)
2080
2081 variant := ctx.ModuleForTests("foo", "android_common")
2082 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2083 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2084 if len(matches) != 2 {
2085 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2086 }
2087 if test.expected != matches[1] {
2088 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2089 }
2090 }
2091}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002092
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002093func TestAndroidAppImport_Filename(t *testing.T) {
2094 ctx, config := testJava(t, `
2095 android_app_import {
2096 name: "foo",
2097 apk: "prebuilts/apk/app.apk",
2098 presigned: true,
2099 }
2100
2101 android_app_import {
2102 name: "bar",
2103 apk: "prebuilts/apk/app.apk",
2104 presigned: true,
2105 filename: "bar_sample.apk"
2106 }
2107 `)
2108
2109 testCases := []struct {
2110 name string
2111 expected string
2112 }{
2113 {
2114 name: "foo",
2115 expected: "foo.apk",
2116 },
2117 {
2118 name: "bar",
2119 expected: "bar_sample.apk",
2120 },
2121 }
2122
2123 for _, test := range testCases {
2124 variant := ctx.ModuleForTests(test.name, "android_common")
2125 if variant.MaybeOutput(test.expected).Rule == nil {
2126 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2127 }
2128
2129 a := variant.Module().(*AndroidAppImport)
2130 expectedValues := []string{test.expected}
2131 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002132 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002133 if !reflect.DeepEqual(actualValues, expectedValues) {
2134 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2135 actualValues, expectedValues)
2136 }
2137 }
2138}
2139
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002140func TestAndroidAppImport_ArchVariants(t *testing.T) {
2141 // The test config's target arch is ARM64.
2142 testCases := []struct {
2143 name string
2144 bp string
2145 expected string
2146 }{
2147 {
2148 name: "matching arch",
2149 bp: `
2150 android_app_import {
2151 name: "foo",
2152 apk: "prebuilts/apk/app.apk",
2153 arch: {
2154 arm64: {
2155 apk: "prebuilts/apk/app_arm64.apk",
2156 },
2157 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002158 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002159 dex_preopt: {
2160 enabled: true,
2161 },
2162 }
2163 `,
2164 expected: "prebuilts/apk/app_arm64.apk",
2165 },
2166 {
2167 name: "no matching arch",
2168 bp: `
2169 android_app_import {
2170 name: "foo",
2171 apk: "prebuilts/apk/app.apk",
2172 arch: {
2173 arm: {
2174 apk: "prebuilts/apk/app_arm.apk",
2175 },
2176 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002177 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002178 dex_preopt: {
2179 enabled: true,
2180 },
2181 }
2182 `,
2183 expected: "prebuilts/apk/app.apk",
2184 },
2185 }
2186
2187 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2188 for _, test := range testCases {
2189 ctx, _ := testJava(t, test.bp)
2190
2191 variant := ctx.ModuleForTests("foo", "android_common")
2192 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2193 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2194 if len(matches) != 2 {
2195 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2196 }
2197 if test.expected != matches[1] {
2198 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2199 }
2200 }
2201}
2202
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002203func TestAndroidTestImport(t *testing.T) {
2204 ctx, config := testJava(t, `
2205 android_test_import {
2206 name: "foo",
2207 apk: "prebuilts/apk/app.apk",
2208 presigned: true,
2209 data: [
2210 "testdata/data",
2211 ],
2212 }
2213 `)
2214
2215 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2216
2217 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002218 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002219 expected := []string{"tests"}
2220 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2221 if !reflect.DeepEqual(expected, actual) {
2222 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2223 }
2224 expected = []string{"testdata/data:testdata/data"}
2225 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2226 if !reflect.DeepEqual(expected, actual) {
2227 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2228 }
2229}
2230
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002231func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2232 ctx, _ := testJava(t, `
2233 android_test_import {
2234 name: "foo",
2235 apk: "prebuilts/apk/app.apk",
2236 certificate: "cert/new_cert",
2237 data: [
2238 "testdata/data",
2239 ],
2240 }
2241
2242 android_test_import {
2243 name: "foo_presigned",
2244 apk: "prebuilts/apk/app.apk",
2245 presigned: true,
2246 data: [
2247 "testdata/data",
2248 ],
2249 }
2250 `)
2251
2252 variant := ctx.ModuleForTests("foo", "android_common")
2253 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2254 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2255 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2256 }
2257
2258 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2259 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2260 if jniRule != android.Cp.String() {
2261 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2262 }
2263}
2264
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002265func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002266 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002267 cc_library {
2268 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002269 sdk_version: "current",
2270 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002271 }
2272
2273 android_test {
2274 name: "stl",
2275 jni_libs: ["libjni"],
2276 compile_multilib: "both",
2277 sdk_version: "current",
2278 stl: "c++_shared",
2279 }
2280
2281 android_test {
2282 name: "system",
2283 jni_libs: ["libjni"],
2284 compile_multilib: "both",
2285 sdk_version: "current",
2286 }
2287 `)
2288
2289 testCases := []struct {
2290 name string
2291 jnis []string
2292 }{
2293 {"stl",
2294 []string{
2295 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002296 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002297 },
2298 },
2299 {"system",
2300 []string{
2301 "libjni.so",
2302 },
2303 },
2304 }
2305
2306 for _, test := range testCases {
2307 t.Run(test.name, func(t *testing.T) {
2308 app := ctx.ModuleForTests(test.name, "android_common")
2309 jniLibZip := app.Output("jnilibs.zip")
2310 var jnis []string
2311 args := strings.Fields(jniLibZip.Args["jarArgs"])
2312 for i := 0; i < len(args); i++ {
2313 if args[i] == "-f" {
2314 jnis = append(jnis, args[i+1])
2315 i += 1
2316 }
2317 }
2318 jnisJoined := strings.Join(jnis, " ")
2319 for _, jni := range test.jnis {
2320 if !strings.Contains(jnisJoined, jni) {
2321 t.Errorf("missing jni %q in %q", jni, jnis)
2322 }
2323 }
2324 })
2325 }
2326}
Colin Cross50ddcc42019-05-16 12:28:22 -07002327
2328func TestUsesLibraries(t *testing.T) {
2329 bp := `
2330 java_sdk_library {
2331 name: "foo",
2332 srcs: ["a.java"],
2333 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002334 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002335 }
2336
2337 java_sdk_library {
2338 name: "bar",
2339 srcs: ["a.java"],
2340 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002341 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002342 }
2343
2344 android_app {
2345 name: "app",
2346 srcs: ["a.java"],
2347 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002348 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002349 optional_uses_libs: [
2350 "bar",
2351 "baz",
2352 ],
2353 }
2354
2355 android_app_import {
2356 name: "prebuilt",
2357 apk: "prebuilts/apk/app.apk",
2358 certificate: "platform",
2359 uses_libs: ["foo"],
2360 optional_uses_libs: [
2361 "bar",
2362 "baz",
2363 ],
2364 }
2365 `
2366
Colin Cross98be1bb2019-12-13 20:41:13 -08002367 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002368 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2369
Colin Cross98be1bb2019-12-13 20:41:13 -08002370 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002371
2372 run(t, ctx, config)
2373
2374 app := ctx.ModuleForTests("app", "android_common")
2375 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2376
2377 // Test that all libraries are verified
2378 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2379 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2380 t.Errorf("wanted %q in %q", w, cmd)
2381 }
2382
2383 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2384 t.Errorf("wanted %q in %q", w, cmd)
2385 }
2386
2387 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2388
2389 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2390 t.Errorf("wanted %q in %q", w, cmd)
2391 }
2392
2393 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2394 t.Errorf("wanted %q in %q", w, cmd)
2395 }
2396
2397 // Test that only present libraries are preopted
2398 cmd = app.Rule("dexpreopt").RuleParams.Command
2399
2400 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2401 t.Errorf("wanted %q in %q", w, cmd)
2402 }
2403
2404 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2405
2406 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2407 t.Errorf("wanted %q in %q", w, cmd)
2408 }
2409}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002410
2411func TestCodelessApp(t *testing.T) {
2412 testCases := []struct {
2413 name string
2414 bp string
2415 noCode bool
2416 }{
2417 {
2418 name: "normal",
2419 bp: `
2420 android_app {
2421 name: "foo",
2422 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002423 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002424 }
2425 `,
2426 noCode: false,
2427 },
2428 {
2429 name: "app without sources",
2430 bp: `
2431 android_app {
2432 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002433 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002434 }
2435 `,
2436 noCode: true,
2437 },
2438 {
2439 name: "app with libraries",
2440 bp: `
2441 android_app {
2442 name: "foo",
2443 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002444 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002445 }
2446
2447 java_library {
2448 name: "lib",
2449 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002450 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002451 }
2452 `,
2453 noCode: false,
2454 },
2455 {
2456 name: "app with sourceless libraries",
2457 bp: `
2458 android_app {
2459 name: "foo",
2460 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002461 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002462 }
2463
2464 java_library {
2465 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002466 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002467 }
2468 `,
2469 // TODO(jungjw): this should probably be true
2470 noCode: false,
2471 },
2472 }
2473
2474 for _, test := range testCases {
2475 t.Run(test.name, func(t *testing.T) {
2476 ctx := testApp(t, test.bp)
2477
2478 foo := ctx.ModuleForTests("foo", "android_common")
2479 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2480 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2481 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2482 }
2483 })
2484 }
2485}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002486
2487func TestEmbedNotice(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002488 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002489 android_app {
2490 name: "foo",
2491 srcs: ["a.java"],
2492 static_libs: ["javalib"],
2493 jni_libs: ["libjni"],
2494 notice: "APP_NOTICE",
2495 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002496 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002497 }
2498
2499 // No embed_notice flag
2500 android_app {
2501 name: "bar",
2502 srcs: ["a.java"],
2503 jni_libs: ["libjni"],
2504 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002505 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002506 }
2507
2508 // No NOTICE files
2509 android_app {
2510 name: "baz",
2511 srcs: ["a.java"],
2512 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002513 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002514 }
2515
2516 cc_library {
2517 name: "libjni",
2518 system_shared_libs: [],
2519 stl: "none",
2520 notice: "LIB_NOTICE",
Colin Cross094cde42020-02-15 10:38:00 -08002521 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002522 }
2523
2524 java_library {
2525 name: "javalib",
2526 srcs: [
2527 ":gen",
2528 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002529 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002530 }
2531
2532 genrule {
2533 name: "gen",
2534 tools: ["gentool"],
2535 out: ["gen.java"],
2536 notice: "GENRULE_NOTICE",
2537 }
2538
2539 java_binary_host {
2540 name: "gentool",
2541 srcs: ["b.java"],
2542 notice: "TOOL_NOTICE",
2543 }
2544 `)
2545
2546 // foo has NOTICE files to process, and embed_notices is true.
2547 foo := ctx.ModuleForTests("foo", "android_common")
2548 // verify merge notices rule.
2549 mergeNotices := foo.Rule("mergeNoticesRule")
2550 noticeInputs := mergeNotices.Inputs.Strings()
2551 // TOOL_NOTICE should be excluded as it's a host module.
2552 if len(mergeNotices.Inputs) != 3 {
2553 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2554 }
2555 if !inList("APP_NOTICE", noticeInputs) {
2556 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2557 }
2558 if !inList("LIB_NOTICE", noticeInputs) {
2559 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2560 }
2561 if !inList("GENRULE_NOTICE", noticeInputs) {
2562 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2563 }
2564 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2565 res := foo.Output("package-res.apk")
2566 aapt2Flags := res.Args["flags"]
2567 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2568 if !strings.Contains(aapt2Flags, e) {
2569 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2570 }
2571
2572 // bar has NOTICE files to process, but embed_notices is not set.
2573 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002574 res = bar.Output("package-res.apk")
2575 aapt2Flags = res.Args["flags"]
2576 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2577 if strings.Contains(aapt2Flags, e) {
2578 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002579 }
2580
2581 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2582 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002583 res = baz.Output("package-res.apk")
2584 aapt2Flags = res.Args["flags"]
2585 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2586 if strings.Contains(aapt2Flags, e) {
2587 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002588 }
2589}
Colin Cross53a87f52019-06-25 13:35:30 -07002590
2591func TestUncompressDex(t *testing.T) {
2592 testCases := []struct {
2593 name string
2594 bp string
2595
2596 uncompressedPlatform bool
2597 uncompressedUnbundled bool
2598 }{
2599 {
2600 name: "normal",
2601 bp: `
2602 android_app {
2603 name: "foo",
2604 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002605 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002606 }
2607 `,
2608 uncompressedPlatform: true,
2609 uncompressedUnbundled: false,
2610 },
2611 {
2612 name: "use_embedded_dex",
2613 bp: `
2614 android_app {
2615 name: "foo",
2616 use_embedded_dex: true,
2617 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002618 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002619 }
2620 `,
2621 uncompressedPlatform: true,
2622 uncompressedUnbundled: true,
2623 },
2624 {
2625 name: "privileged",
2626 bp: `
2627 android_app {
2628 name: "foo",
2629 privileged: true,
2630 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002631 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002632 }
2633 `,
2634 uncompressedPlatform: true,
2635 uncompressedUnbundled: true,
2636 },
2637 }
2638
2639 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2640 t.Helper()
2641
Colin Cross98be1bb2019-12-13 20:41:13 -08002642 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002643 if unbundled {
2644 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2645 }
2646
Colin Cross98be1bb2019-12-13 20:41:13 -08002647 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002648
2649 run(t, ctx, config)
2650
2651 foo := ctx.ModuleForTests("foo", "android_common")
2652 dex := foo.Rule("r8")
2653 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2654 aligned := foo.MaybeRule("zipalign").Rule != nil
2655
2656 if uncompressedInDexJar != want {
2657 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2658 }
2659
2660 if aligned != want {
2661 t.Errorf("want aligned %v, got %v", want, aligned)
2662 }
2663 }
2664
2665 for _, tt := range testCases {
2666 t.Run(tt.name, func(t *testing.T) {
2667 t.Run("platform", func(t *testing.T) {
2668 test(t, tt.bp, tt.uncompressedPlatform, false)
2669 })
2670 t.Run("unbundled", func(t *testing.T) {
2671 test(t, tt.bp, tt.uncompressedUnbundled, true)
2672 })
2673 })
2674 }
2675}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002676
2677func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2678 if expectedValue != "" {
2679 expectedFlag := "--" + flagName + " " + expectedValue
2680 if !strings.Contains(aapt2Flags, expectedFlag) {
2681 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2682 }
2683 } else {
2684 unexpectedFlag := "--" + flagName
2685 if strings.Contains(aapt2Flags, unexpectedFlag) {
2686 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2687 }
2688 }
2689}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002690
2691func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002692 fs := map[string][]byte{
2693 "baz/res/res/values/strings.xml": nil,
2694 "bar/res/res/values/strings.xml": nil,
2695 }
2696 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002697 runtime_resource_overlay {
2698 name: "foo",
2699 certificate: "platform",
2700 product_specific: true,
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002701 static_libs: ["bar"],
2702 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002703 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002704 }
2705
2706 runtime_resource_overlay {
2707 name: "foo_themed",
2708 certificate: "platform",
2709 product_specific: true,
2710 theme: "faza",
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002711 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002712 }
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002713
2714 android_library {
2715 name: "bar",
2716 resource_dirs: ["bar/res"],
2717 }
2718
2719 android_app {
2720 name: "baz",
2721 sdk_version: "current",
2722 resource_dirs: ["baz/res"],
2723 }
2724 `
2725 config := testAppConfig(nil, bp, fs)
2726 ctx := testContext()
2727 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002728
2729 m := ctx.ModuleForTests("foo", "android_common")
2730
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002731 // Check AAPT2 link flags.
2732 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2733 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2734 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2735 if len(absentFlags) > 0 {
2736 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2737 }
2738
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002739 // Check overlay.list output for static_libs dependency.
2740 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
2741 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
2742 if !inList(staticLibPackage, overlayList) {
2743 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
2744 }
2745
2746 // Check AAPT2 link flags for resource_libs dependency.
2747 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
2748 if !strings.Contains(aapt2Flags, resourceLibFlag) {
2749 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
2750 }
2751
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002752 // Check cert signing flag.
2753 signedApk := m.Output("signed/foo.apk")
2754 signingFlag := signedApk.Args["certificates"]
2755 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2756 if expected != signingFlag {
2757 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2758 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002759 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2760 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002761 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
2762 if !reflect.DeepEqual(path, expectedPath) {
2763 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
2764 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002765
2766 // Check device location.
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002767 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002768 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002769 if !reflect.DeepEqual(path, expectedPath) {
2770 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2771 }
2772
2773 // A themed module has a different device location
2774 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002775 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2776 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002777 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
2778 if !reflect.DeepEqual(path, expectedPath) {
2779 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2780 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002781
2782 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
2783 expectedOverrides := []string{"foo"}
2784 if !reflect.DeepEqual(overrides, expectedOverrides) {
2785 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
2786 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002787}
Roshan Pius4df2bc72020-04-27 09:42:27 -07002788
2789func TestOverrideRuntimeResourceOverlay(t *testing.T) {
2790 ctx, _ := testJava(t, `
2791 runtime_resource_overlay {
2792 name: "foo_overlay",
2793 certificate: "platform",
2794 product_specific: true,
2795 sdk_version: "current",
2796 }
2797
2798 override_runtime_resource_overlay {
2799 name: "bar_overlay",
2800 base: "foo_overlay",
2801 package_name: "com.android.bar.overlay",
2802 target_package_name: "com.android.bar",
2803 }
2804 `)
2805
2806 expectedVariants := []struct {
2807 moduleName string
2808 variantName string
2809 apkPath string
2810 overrides []string
2811 targetVariant string
2812 packageFlag string
2813 targetPackageFlag string
2814 }{
2815 {
2816 variantName: "android_common",
2817 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
2818 overrides: nil,
2819 targetVariant: "android_common",
2820 packageFlag: "",
2821 targetPackageFlag: "",
2822 },
2823 {
2824 variantName: "android_common_bar_overlay",
2825 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
2826 overrides: []string{"foo_overlay"},
2827 targetVariant: "android_common_bar",
2828 packageFlag: "com.android.bar.overlay",
2829 targetPackageFlag: "com.android.bar",
2830 },
2831 }
2832 for _, expected := range expectedVariants {
2833 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
2834
2835 // Check the final apk name
2836 outputs := variant.AllOutputs()
2837 expectedApkPath := buildDir + expected.apkPath
2838 found := false
2839 for _, o := range outputs {
2840 if o == expectedApkPath {
2841 found = true
2842 break
2843 }
2844 }
2845 if !found {
2846 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
2847 }
2848
2849 // Check if the overrides field values are correctly aggregated.
2850 mod := variant.Module().(*RuntimeResourceOverlay)
2851 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
2852 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
2853 expected.overrides, mod.properties.Overrides)
2854 }
2855
2856 // Check aapt2 flags.
2857 res := variant.Output("package-res.apk")
2858 aapt2Flags := res.Args["flags"]
2859 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
2860 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
2861 }
2862}
Jaewoong Jungbf135462020-04-26 15:10:51 -07002863
2864func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
2865 ctx, config := testJava(t, `
2866 java_defaults {
2867 name: "rro_defaults",
2868 theme: "default_theme",
2869 product_specific: true,
2870 aaptflags: ["--keep-raw-values"],
2871 }
2872
2873 runtime_resource_overlay {
2874 name: "foo_with_defaults",
2875 defaults: ["rro_defaults"],
2876 }
2877
2878 runtime_resource_overlay {
2879 name: "foo_barebones",
2880 }
2881 `)
2882
2883 //
2884 // RRO module with defaults
2885 //
2886 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
2887
2888 // Check AAPT2 link flags.
2889 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
2890 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2891 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
2892 if len(absentFlags) > 0 {
2893 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2894 }
2895
2896 // Check device location.
2897 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2898 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
2899 if !reflect.DeepEqual(path, expectedPath) {
2900 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
2901 }
2902
2903 //
2904 // RRO module without defaults
2905 //
2906 m = ctx.ModuleForTests("foo_barebones", "android_common")
2907
2908 // Check AAPT2 link flags.
2909 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
2910 unexpectedFlags := "--keep-raw-values"
2911 if inList(unexpectedFlags, aapt2Flags) {
2912 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
2913 }
2914
2915 // Check device location.
2916 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2917 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
2918 if !reflect.DeepEqual(path, expectedPath) {
2919 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2920 }
2921}