blob: 49364f04fd8b77475bdbabd9e839c1af723d0e6c [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"
Colin Crossb69301e2017-12-01 10:48:26 -080021 "sort"
Colin Crossd09b0b62018-04-18 11:06:47 -070022 "strings"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080023 "testing"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070024
25 "github.com/google/blueprint/proptools"
26
27 "android/soong/android"
28 "android/soong/cc"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080029)
30
31var (
32 resourceFiles = []string{
33 "res/layout/layout.xml",
34 "res/values/strings.xml",
35 "res/values-en-rUS/strings.xml",
36 }
37
38 compiledResourceFiles = []string{
39 "aapt2/res/layout_layout.xml.flat",
40 "aapt2/res/values_strings.arsc.flat",
41 "aapt2/res/values-en-rUS_strings.arsc.flat",
42 }
43)
44
Colin Cross98be1bb2019-12-13 20:41:13 -080045func testAppConfig(env map[string]string, bp string, fs map[string][]byte) android.Config {
Colin Cross527012a2017-11-30 22:56:16 -080046 appFS := map[string][]byte{}
47 for k, v := range fs {
48 appFS[k] = v
Colin Cross3bc7ffa2017-11-22 16:19:37 -080049 }
50
Colin Cross527012a2017-11-30 22:56:16 -080051 for _, file := range resourceFiles {
52 appFS[file] = nil
53 }
54
Colin Cross98be1bb2019-12-13 20:41:13 -080055 return testConfig(env, bp, appFS)
Colin Cross527012a2017-11-30 22:56:16 -080056}
57
58func testApp(t *testing.T, bp string) *android.TestContext {
Colin Cross98be1bb2019-12-13 20:41:13 -080059 config := testAppConfig(nil, bp, nil)
Colin Cross527012a2017-11-30 22:56:16 -080060
Colin Crossae8600b2020-10-29 17:09:13 -070061 ctx := testContext(config)
Colin Cross527012a2017-11-30 22:56:16 -080062
63 run(t, ctx, config)
64
65 return ctx
Colin Cross3bc7ffa2017-11-22 16:19:37 -080066}
67
68func TestApp(t *testing.T) {
Colin Crossa97c5d32018-03-28 14:58:31 -070069 for _, moduleType := range []string{"android_app", "android_library"} {
70 t.Run(moduleType, func(t *testing.T) {
71 ctx := testApp(t, moduleType+` {
72 name: "foo",
73 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +090074 sdk_version: "current"
Colin Crossa97c5d32018-03-28 14:58:31 -070075 }
76 `)
Colin Cross3bc7ffa2017-11-22 16:19:37 -080077
Colin Crossa97c5d32018-03-28 14:58:31 -070078 foo := ctx.ModuleForTests("foo", "android_common")
Colin Cross3bc7ffa2017-11-22 16:19:37 -080079
Colin Cross31656952018-05-24 16:11:20 -070080 var expectedLinkImplicits []string
81
82 manifestFixer := foo.Output("manifest_fixer/AndroidManifest.xml")
83 expectedLinkImplicits = append(expectedLinkImplicits, manifestFixer.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080084
Colin Crossa97c5d32018-03-28 14:58:31 -070085 frameworkRes := ctx.ModuleForTests("framework-res", "android_common")
86 expectedLinkImplicits = append(expectedLinkImplicits,
87 frameworkRes.Output("package-res.apk").Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080088
Colin Crossa97c5d32018-03-28 14:58:31 -070089 // Test the mapping from input files to compiled output file names
90 compile := foo.Output(compiledResourceFiles[0])
91 if !reflect.DeepEqual(resourceFiles, compile.Inputs.Strings()) {
92 t.Errorf("expected aapt2 compile inputs expected:\n %#v\n got:\n %#v",
93 resourceFiles, compile.Inputs.Strings())
94 }
Colin Crossb69301e2017-12-01 10:48:26 -080095
Colin Crossa97c5d32018-03-28 14:58:31 -070096 compiledResourceOutputs := compile.Outputs.Strings()
97 sort.Strings(compiledResourceOutputs)
Colin Crossb69301e2017-12-01 10:48:26 -080098
Colin Crossa97c5d32018-03-28 14:58:31 -070099 expectedLinkImplicits = append(expectedLinkImplicits, compiledResourceOutputs...)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800100
Colin Crossa97c5d32018-03-28 14:58:31 -0700101 list := foo.Output("aapt2/res.list")
102 expectedLinkImplicits = append(expectedLinkImplicits, list.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800103
Colin Crossa97c5d32018-03-28 14:58:31 -0700104 // Check that the link rule uses
105 res := ctx.ModuleForTests("foo", "android_common").Output("package-res.apk")
106 if !reflect.DeepEqual(expectedLinkImplicits, res.Implicits.Strings()) {
107 t.Errorf("expected aapt2 link implicits expected:\n %#v\n got:\n %#v",
108 expectedLinkImplicits, res.Implicits.Strings())
109 }
110 })
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800111 }
112}
Colin Cross890ff552017-11-30 20:13:19 -0800113
Colin Crosse560c4a2019-03-19 16:03:11 -0700114func TestAppSplits(t *testing.T) {
115 ctx := testApp(t, `
116 android_app {
117 name: "foo",
118 srcs: ["a.java"],
119 package_splits: ["v4", "v7,hdpi"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900120 sdk_version: "current"
Colin Crosse560c4a2019-03-19 16:03:11 -0700121 }`)
122
123 foo := ctx.ModuleForTests("foo", "android_common")
124
125 expectedOutputs := []string{
126 filepath.Join(buildDir, ".intermediates/foo/android_common/foo.apk"),
127 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v4.apk"),
128 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v7_hdpi.apk"),
129 }
130 for _, expectedOutput := range expectedOutputs {
131 foo.Output(expectedOutput)
132 }
133
Colin Cross41955e82019-05-29 14:40:35 -0700134 outputFiles, err := foo.Module().(*AndroidApp).OutputFiles("")
135 if err != nil {
136 t.Fatal(err)
137 }
138 if g, w := outputFiles.Strings(), expectedOutputs; !reflect.DeepEqual(g, w) {
139 t.Errorf(`want OutputFiles("") = %q, got %q`, w, g)
Colin Crosse560c4a2019-03-19 16:03:11 -0700140 }
141}
142
Jeongik Cha538c0d02019-07-11 15:54:27 +0900143func TestPlatformAPIs(t *testing.T) {
144 testJava(t, `
145 android_app {
146 name: "foo",
147 srcs: ["a.java"],
148 platform_apis: true,
149 }
150 `)
151
152 testJava(t, `
153 android_app {
154 name: "foo",
155 srcs: ["a.java"],
156 sdk_version: "current",
157 }
158 `)
159
160 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
161 android_app {
162 name: "bar",
163 srcs: ["b.java"],
164 }
165 `)
166
167 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
168 android_app {
169 name: "bar",
170 srcs: ["b.java"],
171 sdk_version: "system_current",
172 platform_apis: true,
173 }
174 `)
175}
176
Jeongik Chae403e9e2019-12-07 00:16:24 +0900177func TestAndroidAppLinkType(t *testing.T) {
178 testJava(t, `
179 android_app {
180 name: "foo",
181 srcs: ["a.java"],
182 libs: ["bar"],
183 static_libs: ["baz"],
184 platform_apis: true,
185 }
186
187 java_library {
188 name: "bar",
189 sdk_version: "current",
190 srcs: ["b.java"],
191 }
192
193 android_library {
194 name: "baz",
195 sdk_version: "system_current",
196 srcs: ["c.java"],
197 }
198 `)
199
Steven Moreland00298982020-11-17 21:44:36 +0000200 testJavaError(t, "consider adjusting sdk_version: OR platform_apis:", `
Jeongik Chae403e9e2019-12-07 00:16:24 +0900201 android_app {
202 name: "foo",
203 srcs: ["a.java"],
204 libs: ["bar"],
205 sdk_version: "current",
206 static_libs: ["baz"],
207 }
208
209 java_library {
210 name: "bar",
211 sdk_version: "current",
212 srcs: ["b.java"],
213 }
214
215 android_library {
216 name: "baz",
217 sdk_version: "system_current",
218 srcs: ["c.java"],
219 }
220 `)
221
222 testJava(t, `
223 android_app {
224 name: "foo",
225 srcs: ["a.java"],
226 libs: ["bar"],
227 sdk_version: "system_current",
228 static_libs: ["baz"],
229 }
230
231 java_library {
232 name: "bar",
233 sdk_version: "current",
234 srcs: ["b.java"],
235 }
236
237 android_library {
238 name: "baz",
239 sdk_version: "system_current",
240 srcs: ["c.java"],
241 }
242 `)
243
Steven Moreland00298982020-11-17 21:44:36 +0000244 testJavaError(t, "consider adjusting sdk_version: OR platform_apis:", `
Jeongik Chae403e9e2019-12-07 00:16:24 +0900245 android_app {
246 name: "foo",
247 srcs: ["a.java"],
248 libs: ["bar"],
249 sdk_version: "system_current",
250 static_libs: ["baz"],
251 }
252
253 java_library {
254 name: "bar",
255 sdk_version: "current",
256 srcs: ["b.java"],
257 }
258
259 android_library {
260 name: "baz",
261 srcs: ["c.java"],
262 }
263 `)
264}
265
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100266func TestUpdatableApps(t *testing.T) {
267 testCases := []struct {
268 name string
269 bp string
270 expectedError string
271 }{
272 {
273 name: "Stable public SDK",
274 bp: `android_app {
275 name: "foo",
276 srcs: ["a.java"],
277 sdk_version: "29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100278 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100279 updatable: true,
280 }`,
281 },
282 {
283 name: "Stable system SDK",
284 bp: `android_app {
285 name: "foo",
286 srcs: ["a.java"],
287 sdk_version: "system_29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100288 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100289 updatable: true,
290 }`,
291 },
292 {
293 name: "Current public SDK",
294 bp: `android_app {
295 name: "foo",
296 srcs: ["a.java"],
297 sdk_version: "current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100298 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100299 updatable: true,
300 }`,
301 },
302 {
303 name: "Current system SDK",
304 bp: `android_app {
305 name: "foo",
306 srcs: ["a.java"],
307 sdk_version: "system_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100308 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100309 updatable: true,
310 }`,
311 },
312 {
313 name: "Current module SDK",
314 bp: `android_app {
315 name: "foo",
316 srcs: ["a.java"],
317 sdk_version: "module_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100318 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100319 updatable: true,
320 }`,
321 },
322 {
323 name: "Current core SDK",
324 bp: `android_app {
325 name: "foo",
326 srcs: ["a.java"],
327 sdk_version: "core_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100328 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100329 updatable: true,
330 }`,
331 },
332 {
333 name: "No Platform APIs",
334 bp: `android_app {
335 name: "foo",
336 srcs: ["a.java"],
337 platform_apis: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100338 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100339 updatable: true,
340 }`,
341 expectedError: "Updatable apps must use stable SDKs",
342 },
343 {
344 name: "No Core Platform APIs",
345 bp: `android_app {
346 name: "foo",
347 srcs: ["a.java"],
348 sdk_version: "core_platform",
Artur Satayevf40fc852020-04-16 13:43:02 +0100349 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100350 updatable: true,
351 }`,
352 expectedError: "Updatable apps must use stable SDKs",
353 },
354 {
355 name: "No unspecified APIs",
356 bp: `android_app {
357 name: "foo",
358 srcs: ["a.java"],
359 updatable: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100360 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100361 }`,
362 expectedError: "Updatable apps must use stable SDK",
363 },
Artur Satayevf40fc852020-04-16 13:43:02 +0100364 {
365 name: "Must specify min_sdk_version",
366 bp: `android_app {
367 name: "app_without_min_sdk_version",
368 srcs: ["a.java"],
369 sdk_version: "29",
370 updatable: true,
371 }`,
372 expectedError: "updatable apps must set min_sdk_version.",
373 },
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100374 }
375
376 for _, test := range testCases {
377 t.Run(test.name, func(t *testing.T) {
378 if test.expectedError == "" {
379 testJava(t, test.bp)
380 } else {
381 testJavaError(t, test.expectedError, test.bp)
382 }
383 })
384 }
385}
386
Jooyung Han749dc692020-04-15 11:03:39 +0900387func TestUpdatableApps_TransitiveDepsShouldSetMinSdkVersion(t *testing.T) {
388 testJavaError(t, `module "bar".*: should support min_sdk_version\(29\)`, cc.GatherRequiredDepsForTest(android.Android)+`
389 android_app {
390 name: "foo",
391 srcs: ["a.java"],
392 updatable: true,
393 sdk_version: "current",
394 min_sdk_version: "29",
395 static_libs: ["bar"],
396 }
397
398 java_library {
399 name: "bar",
400 sdk_version: "current",
401 }
402 `)
403}
404
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900405func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
406 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
407 android_app {
408 name: "foo",
409 srcs: ["a.java"],
410 updatable: true,
411 sdk_version: "current",
412 min_sdk_version: "current",
413 jni_libs: ["libjni"],
414 }
415
416 cc_library {
417 name: "libjni",
418 stl: "none",
419 system_shared_libs: [],
420 sdk_version: "current",
421 }
422 `)
423}
424
425func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
426 bp := cc.GatherRequiredDepsForTest(android.Android) + `
427 android_app {
428 name: "foo",
429 srcs: ["a.java"],
430 updatable: true,
431 sdk_version: "current",
432 min_sdk_version: "29",
433 jni_libs: ["libjni"],
434 }
435
436 cc_library {
437 name: "libjni",
438 stl: "none",
439 system_shared_libs: [],
440 sdk_version: "29",
441 }
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900442 `
443 fs := map[string][]byte{
444 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
445 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
446 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
447 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
448 }
449
450 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
451
452 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
453 var crtbeginFound, crtendFound bool
Dan Albert92fe7402020-07-15 13:33:30 -0700454 expectedCrtBegin := ctx.ModuleForTests("crtbegin_so",
455 "android_arm64_armv8-a_sdk_29").Rule("partialLd").Output
456 expectedCrtEnd := ctx.ModuleForTests("crtend_so",
457 "android_arm64_armv8-a_sdk_29").Rule("partialLd").Output
458 implicits := []string{}
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900459 for _, input := range inputs {
Dan Albert92fe7402020-07-15 13:33:30 -0700460 implicits = append(implicits, input.String())
461 if strings.HasSuffix(input.String(), expectedCrtBegin.String()) {
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900462 crtbeginFound = true
Dan Albert92fe7402020-07-15 13:33:30 -0700463 } else if strings.HasSuffix(input.String(), expectedCrtEnd.String()) {
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900464 crtendFound = true
465 }
466 }
Dan Albert92fe7402020-07-15 13:33:30 -0700467 if !crtbeginFound {
468 t.Error(fmt.Sprintf(
469 "expected implicit with suffix %q, have the following implicits:\n%s",
470 expectedCrtBegin, strings.Join(implicits, "\n")))
471 }
472 if !crtendFound {
473 t.Error(fmt.Sprintf(
474 "expected implicit with suffix %q, have the following implicits:\n%s",
475 expectedCrtEnd, strings.Join(implicits, "\n")))
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900476 }
477}
478
479func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
480 bp := cc.GatherRequiredDepsForTest(android.Android) + `
481 android_app {
482 name: "foo",
483 srcs: ["a.java"],
484 updatable: true,
485 sdk_version: "current",
486 min_sdk_version: "29", // this APK should support 29
487 jni_libs: ["libjni"],
488 }
489
490 cc_library {
491 name: "libjni",
492 stl: "none",
493 sdk_version: "current",
494 }
495 `
496 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
497}
498
499func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
500 bp := cc.GatherRequiredDepsForTest(android.Android) + `
501 android_app {
502 name: "foo",
503 srcs: ["a.java"],
504 updatable: true,
505 sdk_version: "current",
506 min_sdk_version: "29", // this APK should support 29
507 jni_libs: ["libjni"],
508 }
509
510 cc_library {
511 name: "libjni",
512 stl: "none",
513 shared_libs: ["libbar"],
514 system_shared_libs: [],
515 sdk_version: "27",
516 }
517
518 cc_library {
519 name: "libbar",
520 stl: "none",
521 system_shared_libs: [],
522 sdk_version: "current",
523 }
524 `
525 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
526}
527
Colin Cross0ddae7f2019-02-07 15:30:01 -0800528func TestResourceDirs(t *testing.T) {
529 testCases := []struct {
530 name string
531 prop string
532 resources []string
533 }{
534 {
535 name: "no resource_dirs",
536 prop: "",
537 resources: []string{"res/res/values/strings.xml"},
538 },
539 {
540 name: "resource_dirs",
541 prop: `resource_dirs: ["res"]`,
542 resources: []string{"res/res/values/strings.xml"},
543 },
544 {
545 name: "empty resource_dirs",
546 prop: `resource_dirs: []`,
547 resources: nil,
548 },
549 }
550
551 fs := map[string][]byte{
552 "res/res/values/strings.xml": nil,
553 }
554
555 bp := `
556 android_app {
557 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900558 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800559 %s
560 }
561 `
562
563 for _, testCase := range testCases {
564 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800565 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
Colin Crossae8600b2020-10-29 17:09:13 -0700566 ctx := testContext(config)
Colin Cross0ddae7f2019-02-07 15:30:01 -0800567 run(t, ctx, config)
568
569 module := ctx.ModuleForTests("foo", "android_common")
570 resourceList := module.MaybeOutput("aapt2/res.list")
571
572 var resources []string
573 if resourceList.Rule != nil {
574 for _, compiledResource := range resourceList.Inputs.Strings() {
575 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
576 }
577 }
578
579 if !reflect.DeepEqual(resources, testCase.resources) {
580 t.Errorf("expected resource files %q, got %q",
581 testCase.resources, resources)
582 }
583 })
584 }
585}
586
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800587func TestLibraryAssets(t *testing.T) {
588 bp := `
589 android_app {
590 name: "foo",
591 sdk_version: "current",
592 static_libs: ["lib1", "lib2", "lib3"],
593 }
594
595 android_library {
596 name: "lib1",
597 sdk_version: "current",
598 asset_dirs: ["assets_a"],
599 }
600
601 android_library {
602 name: "lib2",
603 sdk_version: "current",
604 }
605
606 android_library {
607 name: "lib3",
608 sdk_version: "current",
609 static_libs: ["lib4"],
610 }
611
612 android_library {
613 name: "lib4",
614 sdk_version: "current",
615 asset_dirs: ["assets_b"],
616 }
617 `
618
619 testCases := []struct {
620 name string
621 assetFlag string
622 assetPackages []string
623 }{
624 {
625 name: "foo",
626 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
627 assetPackages: []string{
628 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
629 buildDir + "/.intermediates/lib1/android_common/assets.zip",
630 buildDir + "/.intermediates/lib3/android_common/assets.zip",
631 },
632 },
633 {
634 name: "lib1",
635 assetFlag: "-A assets_a",
636 },
637 {
638 name: "lib2",
639 },
640 {
641 name: "lib3",
642 assetPackages: []string{
643 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
644 buildDir + "/.intermediates/lib4/android_common/assets.zip",
645 },
646 },
647 {
648 name: "lib4",
649 assetFlag: "-A assets_b",
650 },
651 }
652 ctx := testApp(t, bp)
653
654 for _, test := range testCases {
655 t.Run(test.name, func(t *testing.T) {
656 m := ctx.ModuleForTests(test.name, "android_common")
657
658 // Check asset flag in aapt2 link flags
659 var aapt2link android.TestingBuildParams
660 if len(test.assetPackages) > 0 {
661 aapt2link = m.Output("aapt2/package-res.apk")
662 } else {
663 aapt2link = m.Output("package-res.apk")
664 }
665 aapt2Flags := aapt2link.Args["flags"]
666 if test.assetFlag != "" {
667 if !strings.Contains(aapt2Flags, test.assetFlag) {
668 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
669 }
670 } else {
671 if strings.Contains(aapt2Flags, " -A ") {
672 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
673 }
674 }
675
676 // Check asset merge rule.
677 if len(test.assetPackages) > 0 {
678 mergeAssets := m.Output("package-res.apk")
679 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
680 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
681 mergeAssets.Inputs.Strings(), test.assetPackages)
682 }
683 }
684 })
685 }
686}
687
Colin Crossbec85302019-02-13 13:15:46 -0800688func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800689 testCases := []struct {
690 name string
691 enforceRROTargets []string
692 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800693 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800694 overlayFiles map[string][]string
695 rroDirs map[string][]string
696 }{
697 {
698 name: "no RRO",
699 enforceRROTargets: nil,
700 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800701 resourceFiles: map[string][]string{
702 "foo": nil,
703 "bar": {"bar/res/res/values/strings.xml"},
704 "lib": nil,
705 "lib2": {"lib2/res/res/values/strings.xml"},
706 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800707 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800708 "foo": {
709 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800710 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000711 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800712 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800713 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
714 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000715 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800716 },
Colin Crossbec85302019-02-13 13:15:46 -0800717 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800718 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
719 "device/vendor/blah/overlay/bar/res/values/strings.xml",
720 },
Colin Crossbec85302019-02-13 13:15:46 -0800721 "lib": {
722 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
723 "lib/res/res/values/strings.xml",
724 "device/vendor/blah/overlay/lib/res/values/strings.xml",
725 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800726 },
727 rroDirs: map[string][]string{
728 "foo": nil,
729 "bar": nil,
730 },
731 },
732 {
733 name: "enforce RRO on foo",
734 enforceRROTargets: []string{"foo"},
735 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800736 resourceFiles: map[string][]string{
737 "foo": nil,
738 "bar": {"bar/res/res/values/strings.xml"},
739 "lib": nil,
740 "lib2": {"lib2/res/res/values/strings.xml"},
741 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800742 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800743 "foo": {
744 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800745 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000746 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800747 "foo/res/res/values/strings.xml",
748 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
749 },
Colin Crossbec85302019-02-13 13:15:46 -0800750 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800751 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
752 "device/vendor/blah/overlay/bar/res/values/strings.xml",
753 },
Colin Crossbec85302019-02-13 13:15:46 -0800754 "lib": {
755 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
756 "lib/res/res/values/strings.xml",
Colin Crossbec85302019-02-13 13:15:46 -0800757 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800758 },
Colin Crossc1c37552019-01-31 11:42:41 -0800759
Colin Cross5c4791c2019-02-01 11:44:44 -0800760 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800761 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000762 "device:device/vendor/blah/overlay/foo/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000763 "product:product/vendor/blah/overlay/foo/res",
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700764 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800765 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800766 "bar": nil,
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700767 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800768 },
769 },
770 {
771 name: "enforce RRO on all",
772 enforceRROTargets: []string{"*"},
773 enforceRROExcludedOverlays: []string{
774 // Excluding specific apps/res directories also allowed.
775 "device/vendor/blah/static_overlay/foo",
776 "device/vendor/blah/static_overlay/bar/res",
777 },
Colin Crossbec85302019-02-13 13:15:46 -0800778 resourceFiles: map[string][]string{
779 "foo": nil,
780 "bar": {"bar/res/res/values/strings.xml"},
781 "lib": nil,
782 "lib2": {"lib2/res/res/values/strings.xml"},
783 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800784 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800785 "foo": {
786 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800787 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000788 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800789 "foo/res/res/values/strings.xml",
790 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
791 },
Colin Crossbec85302019-02-13 13:15:46 -0800792 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
793 "lib": {
794 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
795 "lib/res/res/values/strings.xml",
796 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800797 },
798 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800799 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000800 "device:device/vendor/blah/overlay/foo/res",
801 "product:product/vendor/blah/overlay/foo/res",
802 // Lib dep comes after the direct deps
803 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800804 },
Anton Hansson53c88442019-03-18 15:53:16 +0000805 "bar": {"device:device/vendor/blah/overlay/bar/res"},
806 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800807 },
808 },
809 }
810
Anton Hansson53c88442019-03-18 15:53:16 +0000811 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800812 "device/vendor/blah/overlay",
813 "device/vendor/blah/overlay2",
814 "device/vendor/blah/static_overlay",
815 }
816
Anton Hansson53c88442019-03-18 15:53:16 +0000817 productResourceOverlays := []string{
818 "product/vendor/blah/overlay",
819 }
820
Colin Cross890ff552017-11-30 20:13:19 -0800821 fs := map[string][]byte{
822 "foo/res/res/values/strings.xml": nil,
823 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800824 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800825 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800826 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
827 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800828 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800829 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
830 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
831 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000832 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800833 }
834
835 bp := `
836 android_app {
837 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900838 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800839 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000840 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800841 }
842
843 android_app {
844 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900845 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800846 resource_dirs: ["bar/res"],
847 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800848
849 android_library {
850 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900851 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800852 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800853 static_libs: ["lib2"],
854 }
855
856 android_library {
857 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900858 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800859 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800860 }
Anton Hansson53c88442019-03-18 15:53:16 +0000861
862 // This library has the same resources as lib (should not lead to dupe RROs)
863 android_library {
864 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900865 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000866 resource_dirs: ["lib/res"]
867 }
Colin Cross890ff552017-11-30 20:13:19 -0800868 `
869
Colin Cross5c4791c2019-02-01 11:44:44 -0800870 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800871 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800872 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000873 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
874 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800875 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800876 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800877 }
878 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800879 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800880 }
881
Colin Crossae8600b2020-10-29 17:09:13 -0700882 ctx := testContext(config)
Colin Cross890ff552017-11-30 20:13:19 -0800883 run(t, ctx, config)
884
Colin Crossbec85302019-02-13 13:15:46 -0800885 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
886 for _, o := range list {
887 res := module.MaybeOutput(o)
888 if res.Rule != nil {
889 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
890 // verify the inputs to the .arsc.flat rule.
891 files = append(files, res.Inputs.Strings()...)
892 } else {
893 // Otherwise, verify the full path to the output of the other module
894 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000895 }
Colin Cross890ff552017-11-30 20:13:19 -0800896 }
Colin Crossbec85302019-02-13 13:15:46 -0800897 return files
Colin Cross890ff552017-11-30 20:13:19 -0800898 }
899
Colin Crossbec85302019-02-13 13:15:46 -0800900 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
901 module := ctx.ModuleForTests(moduleName, "android_common")
902 resourceList := module.MaybeOutput("aapt2/res.list")
903 if resourceList.Rule != nil {
904 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000905 }
Colin Crossbec85302019-02-13 13:15:46 -0800906 overlayList := module.MaybeOutput("aapt2/overlay.list")
907 if overlayList.Rule != nil {
908 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
909 }
910
Anton Hansson53c88442019-03-18 15:53:16 +0000911 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
912 var prefix string
913 if d.overlayType == device {
914 prefix = "device:"
915 } else if d.overlayType == product {
916 prefix = "product:"
917 } else {
918 t.Fatalf("Unexpected overlayType %d", d.overlayType)
919 }
920 rroDirs = append(rroDirs, prefix+d.path.String())
921 }
Colin Crossbec85302019-02-13 13:15:46 -0800922
923 return resourceFiles, overlayFiles, rroDirs
924 }
925
926 modules := []string{"foo", "bar", "lib", "lib2"}
927 for _, module := range modules {
928 resourceFiles, overlayFiles, rroDirs := getResources(module)
929
930 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
931 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
932 module, testCase.resourceFiles[module], resourceFiles)
933 }
934 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
935 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
936 module, testCase.overlayFiles[module], overlayFiles)
937 }
938 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +0000939 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -0800940 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +0000941 }
Colin Cross890ff552017-11-30 20:13:19 -0800942 }
Colin Cross890ff552017-11-30 20:13:19 -0800943 })
944 }
945}
Colin Crossd09b0b62018-04-18 11:06:47 -0700946
Jeongik Cha219141c2020-08-06 23:00:37 +0900947func checkSdkVersion(t *testing.T, config android.Config, expectedSdkVersion string) {
Colin Crossae8600b2020-10-29 17:09:13 -0700948 ctx := testContext(config)
Jeongik Cha219141c2020-08-06 23:00:37 +0900949
950 run(t, ctx, config)
951
952 foo := ctx.ModuleForTests("foo", "android_common")
953 link := foo.Output("package-res.apk")
954 linkFlags := strings.Split(link.Args["flags"], " ")
955 min := android.IndexList("--min-sdk-version", linkFlags)
956 target := android.IndexList("--target-sdk-version", linkFlags)
957
958 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
959 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
960 }
961
962 gotMinSdkVersion := linkFlags[min+1]
963 gotTargetSdkVersion := linkFlags[target+1]
964
965 if gotMinSdkVersion != expectedSdkVersion {
966 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
967 expectedSdkVersion, gotMinSdkVersion)
968 }
969
970 if gotTargetSdkVersion != expectedSdkVersion {
971 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
972 expectedSdkVersion, gotTargetSdkVersion)
973 }
974}
975
Colin Crossd09b0b62018-04-18 11:06:47 -0700976func TestAppSdkVersion(t *testing.T) {
977 testCases := []struct {
978 name string
979 sdkVersion string
980 platformSdkInt int
981 platformSdkCodename string
982 platformSdkFinal bool
983 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +0900984 platformApis bool
Dan Albert4f378d72020-07-23 17:32:15 -0700985 activeCodenames []string
Colin Crossd09b0b62018-04-18 11:06:47 -0700986 }{
987 {
988 name: "current final SDK",
989 sdkVersion: "current",
990 platformSdkInt: 27,
991 platformSdkCodename: "REL",
992 platformSdkFinal: true,
993 expectedMinSdkVersion: "27",
994 },
995 {
996 name: "current non-final SDK",
997 sdkVersion: "current",
998 platformSdkInt: 27,
999 platformSdkCodename: "OMR1",
1000 platformSdkFinal: false,
1001 expectedMinSdkVersion: "OMR1",
Dan Albert4f378d72020-07-23 17:32:15 -07001002 activeCodenames: []string{"OMR1"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001003 },
1004 {
1005 name: "default final SDK",
1006 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001007 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001008 platformSdkInt: 27,
1009 platformSdkCodename: "REL",
1010 platformSdkFinal: true,
1011 expectedMinSdkVersion: "27",
1012 },
1013 {
1014 name: "default non-final SDK",
1015 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001016 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001017 platformSdkInt: 27,
1018 platformSdkCodename: "OMR1",
1019 platformSdkFinal: false,
1020 expectedMinSdkVersion: "OMR1",
Dan Albert4f378d72020-07-23 17:32:15 -07001021 activeCodenames: []string{"OMR1"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001022 },
1023 {
1024 name: "14",
1025 sdkVersion: "14",
1026 expectedMinSdkVersion: "14",
Dan Albert4f378d72020-07-23 17:32:15 -07001027 platformSdkCodename: "S",
1028 activeCodenames: []string{"S"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001029 },
1030 }
1031
1032 for _, moduleType := range []string{"android_app", "android_library"} {
1033 for _, test := range testCases {
1034 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001035 platformApiProp := ""
1036 if test.platformApis {
1037 platformApiProp = "platform_apis: true,"
1038 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001039 bp := fmt.Sprintf(`%s {
1040 name: "foo",
1041 srcs: ["a.java"],
1042 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001043 %s
1044 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001045
Colin Cross98be1bb2019-12-13 20:41:13 -08001046 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001047 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1048 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
Dan Albert4f378d72020-07-23 17:32:15 -07001049 config.TestProductVariables.Platform_version_active_codenames = test.activeCodenames
Colin Crossd09b0b62018-04-18 11:06:47 -07001050 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
Jeongik Cha219141c2020-08-06 23:00:37 +09001051 checkSdkVersion(t, config, test.expectedMinSdkVersion)
Colin Crossd09b0b62018-04-18 11:06:47 -07001052
Colin Crossd09b0b62018-04-18 11:06:47 -07001053 })
1054 }
1055 }
1056}
Colin Crossa4f08812018-10-02 22:03:40 -07001057
Jeongik Cha219141c2020-08-06 23:00:37 +09001058func TestVendorAppSdkVersion(t *testing.T) {
1059 testCases := []struct {
1060 name string
1061 sdkVersion string
1062 platformSdkInt int
1063 platformSdkCodename string
1064 platformSdkFinal bool
1065 deviceCurrentApiLevelForVendorModules string
1066 expectedMinSdkVersion string
1067 }{
1068 {
1069 name: "current final SDK",
1070 sdkVersion: "current",
1071 platformSdkInt: 29,
1072 platformSdkCodename: "REL",
1073 platformSdkFinal: true,
1074 deviceCurrentApiLevelForVendorModules: "29",
1075 expectedMinSdkVersion: "29",
1076 },
1077 {
1078 name: "current final SDK",
1079 sdkVersion: "current",
1080 platformSdkInt: 29,
1081 platformSdkCodename: "REL",
1082 platformSdkFinal: true,
1083 deviceCurrentApiLevelForVendorModules: "28",
1084 expectedMinSdkVersion: "28",
1085 },
1086 {
1087 name: "current final SDK",
1088 sdkVersion: "current",
1089 platformSdkInt: 29,
1090 platformSdkCodename: "Q",
1091 platformSdkFinal: false,
Jeongik Cha219141c2020-08-06 23:00:37 +09001092 deviceCurrentApiLevelForVendorModules: "28",
1093 expectedMinSdkVersion: "28",
1094 },
1095 }
1096
1097 for _, moduleType := range []string{"android_app", "android_library"} {
1098 for _, sdkKind := range []string{"", "system_"} {
1099 for _, test := range testCases {
1100 t.Run(moduleType+" "+test.name, func(t *testing.T) {
1101 bp := fmt.Sprintf(`%s {
1102 name: "foo",
1103 srcs: ["a.java"],
1104 sdk_version: "%s%s",
1105 vendor: true,
1106 }`, moduleType, sdkKind, test.sdkVersion)
1107
1108 config := testAppConfig(nil, bp, nil)
1109 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1110 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1111 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1112 config.TestProductVariables.DeviceCurrentApiLevelForVendorModules = &test.deviceCurrentApiLevelForVendorModules
1113 config.TestProductVariables.DeviceSystemSdkVersions = []string{"28", "29"}
1114 checkSdkVersion(t, config, test.expectedMinSdkVersion)
1115 })
1116 }
1117 }
1118 }
1119}
1120
Paul Duffin50c217c2019-06-12 13:25:22 +01001121func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001122 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001123 cc_library {
1124 name: "libjni",
1125 system_shared_libs: [],
Colin Crossc511bc52020-04-07 16:50:32 +00001126 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001127 stl: "none",
1128 }
1129
1130 android_test {
1131 name: "test",
1132 sdk_version: "core_platform",
1133 jni_libs: ["libjni"],
1134 }
1135
1136 android_test {
1137 name: "test_first",
1138 sdk_version: "core_platform",
1139 compile_multilib: "first",
1140 jni_libs: ["libjni"],
1141 }
1142
1143 android_test {
1144 name: "test_both",
1145 sdk_version: "core_platform",
1146 compile_multilib: "both",
1147 jni_libs: ["libjni"],
1148 }
1149
1150 android_test {
1151 name: "test_32",
1152 sdk_version: "core_platform",
1153 compile_multilib: "32",
1154 jni_libs: ["libjni"],
1155 }
1156
1157 android_test {
1158 name: "test_64",
1159 sdk_version: "core_platform",
1160 compile_multilib: "64",
1161 jni_libs: ["libjni"],
1162 }
1163 `)
1164
1165 testCases := []struct {
1166 name string
1167 abis []string
1168 }{
1169 {"test", []string{"arm64-v8a"}},
1170 {"test_first", []string{"arm64-v8a"}},
1171 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1172 {"test_32", []string{"armeabi-v7a"}},
1173 {"test_64", []string{"arm64-v8a"}},
1174 }
1175
1176 for _, test := range testCases {
1177 t.Run(test.name, func(t *testing.T) {
1178 app := ctx.ModuleForTests(test.name, "android_common")
1179 jniLibZip := app.Output("jnilibs.zip")
1180 var abis []string
1181 args := strings.Fields(jniLibZip.Args["jarArgs"])
1182 for i := 0; i < len(args); i++ {
1183 if args[i] == "-P" {
1184 abis = append(abis, filepath.Base(args[i+1]))
1185 i++
1186 }
1187 }
1188 if !reflect.DeepEqual(abis, test.abis) {
1189 t.Errorf("want abis %v, got %v", test.abis, abis)
1190 }
1191 })
1192 }
1193}
1194
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001195func TestAppSdkVersionByPartition(t *testing.T) {
1196 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1197 android_app {
1198 name: "foo",
1199 srcs: ["a.java"],
1200 vendor: true,
1201 platform_apis: true,
1202 }
1203 `)
1204
1205 testJava(t, `
1206 android_app {
1207 name: "bar",
1208 srcs: ["b.java"],
1209 platform_apis: true,
1210 }
1211 `)
1212
1213 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001214 bp := `
1215 android_app {
1216 name: "foo",
1217 srcs: ["a.java"],
1218 product_specific: true,
1219 platform_apis: true,
1220 }
1221 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001222
1223 config := testAppConfig(nil, bp, nil)
1224 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001225 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001226 testJavaErrorWithConfig(t, "sdk_version must have a value when the module is located at vendor or product", config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001227 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001228 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001229 }
1230 }
1231}
1232
Paul Duffin50c217c2019-06-12 13:25:22 +01001233func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001234 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001235 cc_library {
1236 name: "libjni",
1237 system_shared_libs: [],
1238 stl: "none",
Colin Cross094cde42020-02-15 10:38:00 -08001239 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001240 }
1241
1242 android_app {
1243 name: "app",
1244 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001245 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001246 }
1247
1248 android_app {
1249 name: "app_noembed",
1250 jni_libs: ["libjni"],
1251 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001252 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001253 }
1254
1255 android_app {
1256 name: "app_embed",
1257 jni_libs: ["libjni"],
1258 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001259 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001260 }
1261
1262 android_test {
1263 name: "test",
Colin Crossc511bc52020-04-07 16:50:32 +00001264 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001265 jni_libs: ["libjni"],
1266 }
1267
1268 android_test {
1269 name: "test_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001270 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001271 jni_libs: ["libjni"],
1272 use_embedded_native_libs: false,
1273 }
1274
1275 android_test_helper_app {
1276 name: "test_helper",
Colin Crossc511bc52020-04-07 16:50:32 +00001277 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001278 jni_libs: ["libjni"],
1279 }
1280
1281 android_test_helper_app {
1282 name: "test_helper_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001283 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001284 jni_libs: ["libjni"],
1285 use_embedded_native_libs: false,
1286 }
1287 `)
1288
1289 testCases := []struct {
1290 name string
1291 packaged bool
1292 compressed bool
1293 }{
1294 {"app", false, false},
1295 {"app_noembed", false, false},
1296 {"app_embed", true, false},
1297 {"test", true, false},
1298 {"test_noembed", true, true},
1299 {"test_helper", true, false},
1300 {"test_helper_noembed", true, true},
1301 }
1302
1303 for _, test := range testCases {
1304 t.Run(test.name, func(t *testing.T) {
1305 app := ctx.ModuleForTests(test.name, "android_common")
1306 jniLibZip := app.MaybeOutput("jnilibs.zip")
1307 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1308 t.Errorf("expected jni packaged %v, got %v", w, g)
1309 }
1310
1311 if jniLibZip.Rule != nil {
1312 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1313 t.Errorf("expected jni compressed %v, got %v", w, g)
1314 }
Colin Crossc511bc52020-04-07 16:50:32 +00001315
1316 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1317 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1318 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001319 }
1320 })
1321 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001322}
1323
Colin Cross3c007702020-05-08 11:20:24 -07001324func TestJNISDK(t *testing.T) {
1325 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1326 cc_library {
1327 name: "libjni",
1328 system_shared_libs: [],
1329 stl: "none",
1330 sdk_version: "current",
1331 }
1332
1333 android_test {
1334 name: "app_platform",
1335 jni_libs: ["libjni"],
1336 platform_apis: true,
1337 }
1338
1339 android_test {
1340 name: "app_sdk",
1341 jni_libs: ["libjni"],
1342 sdk_version: "current",
1343 }
1344
1345 android_test {
1346 name: "app_force_platform",
1347 jni_libs: ["libjni"],
1348 sdk_version: "current",
1349 jni_uses_platform_apis: true,
1350 }
1351
1352 android_test {
1353 name: "app_force_sdk",
1354 jni_libs: ["libjni"],
1355 platform_apis: true,
1356 jni_uses_sdk_apis: true,
1357 }
Colin Crossc2d24052020-05-13 11:05:02 -07001358
1359 cc_library {
1360 name: "libvendorjni",
1361 system_shared_libs: [],
1362 stl: "none",
1363 vendor: true,
1364 }
1365
1366 android_test {
1367 name: "app_vendor",
1368 jni_libs: ["libvendorjni"],
1369 sdk_version: "current",
1370 vendor: true,
1371 }
Colin Cross3c007702020-05-08 11:20:24 -07001372 `)
1373
1374 testCases := []struct {
Colin Crossc2d24052020-05-13 11:05:02 -07001375 name string
1376 sdkJNI bool
1377 vendorJNI bool
Colin Cross3c007702020-05-08 11:20:24 -07001378 }{
Colin Crossc2d24052020-05-13 11:05:02 -07001379 {name: "app_platform"},
1380 {name: "app_sdk", sdkJNI: true},
1381 {name: "app_force_platform"},
1382 {name: "app_force_sdk", sdkJNI: true},
1383 {name: "app_vendor", vendorJNI: true},
Colin Cross3c007702020-05-08 11:20:24 -07001384 }
1385
Colin Crossc2d24052020-05-13 11:05:02 -07001386 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1387 Output("libjni.so").Output.String()
1388 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1389 Output("libjni.so").Output.String()
1390 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1391 Output("libvendorjni.so").Output.String()
1392
Colin Cross3c007702020-05-08 11:20:24 -07001393 for _, test := range testCases {
1394 t.Run(test.name, func(t *testing.T) {
1395 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross3c007702020-05-08 11:20:24 -07001396
1397 jniLibZip := app.MaybeOutput("jnilibs.zip")
1398 if len(jniLibZip.Implicits) != 1 {
1399 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1400 }
1401 gotJNI := jniLibZip.Implicits[0].String()
1402
1403 if test.sdkJNI {
1404 if gotJNI != sdkJNI {
1405 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1406 }
Colin Crossc2d24052020-05-13 11:05:02 -07001407 } else if test.vendorJNI {
1408 if gotJNI != vendorJNI {
1409 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1410 }
Colin Cross3c007702020-05-08 11:20:24 -07001411 } else {
1412 if gotJNI != platformJNI {
1413 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1414 }
1415 }
1416 })
1417 }
1418
1419 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1420 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1421 android_test {
1422 name: "app_platform",
1423 platform_apis: true,
1424 jni_uses_platform_apis: true,
1425 }
1426 `)
1427 })
1428
1429 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1430 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1431 android_test {
1432 name: "app_sdk",
1433 sdk_version: "current",
1434 jni_uses_sdk_apis: true,
1435 }
1436 `)
1437 })
1438
1439}
1440
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001441func TestCertificates(t *testing.T) {
1442 testCases := []struct {
1443 name string
1444 bp string
1445 certificateOverride string
Liz Kammere2b27f42020-05-07 13:24:05 -07001446 expectedLineage string
1447 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001448 }{
1449 {
1450 name: "default",
1451 bp: `
1452 android_app {
1453 name: "foo",
1454 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001455 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001456 }
1457 `,
1458 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001459 expectedLineage: "",
1460 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001461 },
1462 {
1463 name: "module certificate property",
1464 bp: `
1465 android_app {
1466 name: "foo",
1467 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001468 certificate: ":new_certificate",
1469 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001470 }
1471
1472 android_app_certificate {
1473 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001474 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001475 }
1476 `,
1477 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001478 expectedLineage: "",
1479 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001480 },
1481 {
1482 name: "path certificate property",
1483 bp: `
1484 android_app {
1485 name: "foo",
1486 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001487 certificate: "expiredkey",
1488 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001489 }
1490 `,
1491 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001492 expectedLineage: "",
1493 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001494 },
1495 {
1496 name: "certificate overrides",
1497 bp: `
1498 android_app {
1499 name: "foo",
1500 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001501 certificate: "expiredkey",
1502 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001503 }
1504
1505 android_app_certificate {
1506 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001507 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001508 }
1509 `,
1510 certificateOverride: "foo:new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001511 expectedLineage: "",
1512 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1513 },
1514 {
1515 name: "certificate lineage",
1516 bp: `
1517 android_app {
1518 name: "foo",
1519 srcs: ["a.java"],
1520 certificate: ":new_certificate",
1521 lineage: "lineage.bin",
1522 sdk_version: "current",
1523 }
1524
1525 android_app_certificate {
1526 name: "new_certificate",
1527 certificate: "cert/new_cert",
1528 }
1529 `,
1530 certificateOverride: "",
1531 expectedLineage: "--lineage lineage.bin",
1532 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001533 },
1534 }
1535
1536 for _, test := range testCases {
1537 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001538 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001539 if test.certificateOverride != "" {
1540 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1541 }
Colin Crossae8600b2020-10-29 17:09:13 -07001542 ctx := testContext(config)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001543
1544 run(t, ctx, config)
1545 foo := ctx.ModuleForTests("foo", "android_common")
1546
1547 signapk := foo.Output("foo.apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001548 signCertificateFlags := signapk.Args["certificates"]
1549 if test.expectedCertificate != signCertificateFlags {
1550 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1551 }
1552
1553 signFlags := signapk.Args["flags"]
1554 if test.expectedLineage != signFlags {
1555 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001556 }
1557 })
1558 }
1559}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001560
Songchun Fan688de9a2020-03-24 20:32:24 -07001561func TestRequestV4SigningFlag(t *testing.T) {
1562 testCases := []struct {
1563 name string
1564 bp string
1565 expected string
1566 }{
1567 {
1568 name: "default",
1569 bp: `
1570 android_app {
1571 name: "foo",
1572 srcs: ["a.java"],
1573 sdk_version: "current",
1574 }
1575 `,
1576 expected: "",
1577 },
1578 {
1579 name: "default",
1580 bp: `
1581 android_app {
1582 name: "foo",
1583 srcs: ["a.java"],
1584 sdk_version: "current",
1585 v4_signature: false,
1586 }
1587 `,
1588 expected: "",
1589 },
1590 {
1591 name: "module certificate property",
1592 bp: `
1593 android_app {
1594 name: "foo",
1595 srcs: ["a.java"],
1596 sdk_version: "current",
1597 v4_signature: true,
1598 }
1599 `,
1600 expected: "--enable-v4",
1601 },
1602 }
1603
1604 for _, test := range testCases {
1605 t.Run(test.name, func(t *testing.T) {
1606 config := testAppConfig(nil, test.bp, nil)
Colin Crossae8600b2020-10-29 17:09:13 -07001607 ctx := testContext(config)
Songchun Fan688de9a2020-03-24 20:32:24 -07001608
1609 run(t, ctx, config)
1610 foo := ctx.ModuleForTests("foo", "android_common")
1611
1612 signapk := foo.Output("foo.apk")
1613 signFlags := signapk.Args["flags"]
1614 if test.expected != signFlags {
1615 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1616 }
1617 })
1618 }
1619}
1620
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001621func TestPackageNameOverride(t *testing.T) {
1622 testCases := []struct {
1623 name string
1624 bp string
1625 packageNameOverride string
1626 expected []string
1627 }{
1628 {
1629 name: "default",
1630 bp: `
1631 android_app {
1632 name: "foo",
1633 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001634 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001635 }
1636 `,
1637 packageNameOverride: "",
1638 expected: []string{
1639 buildDir + "/.intermediates/foo/android_common/foo.apk",
1640 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1641 },
1642 },
1643 {
1644 name: "overridden",
1645 bp: `
1646 android_app {
1647 name: "foo",
1648 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001649 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001650 }
1651 `,
1652 packageNameOverride: "foo:bar",
1653 expected: []string{
1654 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001655 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001656 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1657 },
1658 },
1659 }
1660
1661 for _, test := range testCases {
1662 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001663 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001664 if test.packageNameOverride != "" {
1665 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1666 }
Colin Crossae8600b2020-10-29 17:09:13 -07001667 ctx := testContext(config)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001668
1669 run(t, ctx, config)
1670 foo := ctx.ModuleForTests("foo", "android_common")
1671
1672 outputs := foo.AllOutputs()
1673 outputMap := make(map[string]bool)
1674 for _, o := range outputs {
1675 outputMap[o] = true
1676 }
1677 for _, e := range test.expected {
1678 if _, exist := outputMap[e]; !exist {
1679 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1680 }
1681 }
1682 })
1683 }
1684}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001685
1686func TestInstrumentationTargetOverridden(t *testing.T) {
1687 bp := `
1688 android_app {
1689 name: "foo",
1690 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001691 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001692 }
1693
1694 android_test {
1695 name: "bar",
1696 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001697 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001698 }
1699 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001700 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001701 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Crossae8600b2020-10-29 17:09:13 -07001702 ctx := testContext(config)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001703
1704 run(t, ctx, config)
1705
1706 bar := ctx.ModuleForTests("bar", "android_common")
1707 res := bar.Output("package-res.apk")
1708 aapt2Flags := res.Args["flags"]
1709 e := "--rename-instrumentation-target-package org.dandroid.bp"
1710 if !strings.Contains(aapt2Flags, e) {
1711 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1712 }
1713}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001714
1715func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001716 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001717 android_app {
1718 name: "foo",
1719 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001720 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001721 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001722 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001723 }
1724
1725 override_android_app {
1726 name: "bar",
1727 base: "foo",
1728 certificate: ":new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001729 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001730 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001731 }
1732
1733 android_app_certificate {
1734 name: "new_certificate",
1735 certificate: "cert/new_cert",
1736 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001737
1738 override_android_app {
1739 name: "baz",
1740 base: "foo",
1741 package_name: "org.dandroid.bp",
1742 }
Liz Kammer9f9fd022020-06-18 19:44:06 +00001743
1744 override_android_app {
1745 name: "baz_no_rename_resources",
1746 base: "foo",
1747 package_name: "org.dandroid.bp",
1748 rename_resources_package: false,
1749 }
1750
1751 android_app {
1752 name: "foo_no_rename_resources",
1753 srcs: ["a.java"],
1754 certificate: "expiredkey",
1755 overrides: ["qux"],
1756 rename_resources_package: false,
1757 sdk_version: "current",
1758 }
1759
1760 override_android_app {
1761 name: "baz_base_no_rename_resources",
1762 base: "foo_no_rename_resources",
1763 package_name: "org.dandroid.bp",
1764 }
1765
1766 override_android_app {
1767 name: "baz_override_base_rename_resources",
1768 base: "foo_no_rename_resources",
1769 package_name: "org.dandroid.bp",
1770 rename_resources_package: true,
1771 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001772 `)
1773
1774 expectedVariants := []struct {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001775 name string
1776 moduleName string
1777 variantName string
1778 apkName string
1779 apkPath string
1780 certFlag string
1781 lineageFlag string
1782 overrides []string
1783 packageFlag string
1784 renameResources bool
1785 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001786 }{
1787 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001788 name: "foo",
1789 moduleName: "foo",
1790 variantName: "android_common",
1791 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
1792 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1793 lineageFlag: "",
1794 overrides: []string{"qux"},
1795 packageFlag: "",
1796 renameResources: false,
1797 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001798 },
1799 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001800 name: "foo",
1801 moduleName: "bar",
1802 variantName: "android_common_bar",
1803 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
1804 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1805 lineageFlag: "--lineage lineage.bin",
1806 overrides: []string{"qux", "foo"},
1807 packageFlag: "",
1808 renameResources: false,
1809 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001810 },
1811 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001812 name: "foo",
1813 moduleName: "baz",
1814 variantName: "android_common_baz",
1815 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
1816 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1817 lineageFlag: "",
1818 overrides: []string{"qux", "foo"},
1819 packageFlag: "org.dandroid.bp",
1820 renameResources: true,
1821 logging_parent: "",
1822 },
1823 {
1824 name: "foo",
1825 moduleName: "baz_no_rename_resources",
1826 variantName: "android_common_baz_no_rename_resources",
1827 apkPath: "/target/product/test_device/system/app/baz_no_rename_resources/baz_no_rename_resources.apk",
1828 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1829 lineageFlag: "",
1830 overrides: []string{"qux", "foo"},
1831 packageFlag: "org.dandroid.bp",
1832 renameResources: false,
1833 logging_parent: "",
1834 },
1835 {
1836 name: "foo_no_rename_resources",
1837 moduleName: "baz_base_no_rename_resources",
1838 variantName: "android_common_baz_base_no_rename_resources",
1839 apkPath: "/target/product/test_device/system/app/baz_base_no_rename_resources/baz_base_no_rename_resources.apk",
1840 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1841 lineageFlag: "",
1842 overrides: []string{"qux", "foo_no_rename_resources"},
1843 packageFlag: "org.dandroid.bp",
1844 renameResources: false,
1845 logging_parent: "",
1846 },
1847 {
1848 name: "foo_no_rename_resources",
1849 moduleName: "baz_override_base_rename_resources",
1850 variantName: "android_common_baz_override_base_rename_resources",
1851 apkPath: "/target/product/test_device/system/app/baz_override_base_rename_resources/baz_override_base_rename_resources.apk",
1852 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1853 lineageFlag: "",
1854 overrides: []string{"qux", "foo_no_rename_resources"},
1855 packageFlag: "org.dandroid.bp",
1856 renameResources: true,
1857 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001858 },
1859 }
1860 for _, expected := range expectedVariants {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001861 variant := ctx.ModuleForTests(expected.name, expected.variantName)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001862
1863 // Check the final apk name
1864 outputs := variant.AllOutputs()
1865 expectedApkPath := buildDir + expected.apkPath
1866 found := false
1867 for _, o := range outputs {
1868 if o == expectedApkPath {
1869 found = true
1870 break
1871 }
1872 }
1873 if !found {
1874 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1875 }
1876
1877 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001878 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001879 certFlag := signapk.Args["certificates"]
1880 if expected.certFlag != certFlag {
1881 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1882 }
1883
1884 // Check the lineage flags
1885 lineageFlag := signapk.Args["flags"]
1886 if expected.lineageFlag != lineageFlag {
1887 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001888 }
1889
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001890 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001891 mod := variant.Module().(*AndroidApp)
1892 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1893 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1894 expected.overrides, mod.appProperties.Overrides)
1895 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001896
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001897 // Test Overridable property: Logging_parent
1898 logging_parent := mod.aapt.LoggingParent
1899 if expected.logging_parent != logging_parent {
1900 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1901 expected.logging_parent, logging_parent)
1902 }
1903
Liz Kammer1d5983b2020-05-19 19:15:37 +00001904 // Check the package renaming flag, if exists.
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001905 res := variant.Output("package-res.apk")
1906 aapt2Flags := res.Args["flags"]
Liz Kammer9f9fd022020-06-18 19:44:06 +00001907 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1908 expectedPackage := expected.packageFlag
1909 if !expected.renameResources {
1910 expectedPackage = ""
Liz Kammer1d5983b2020-05-19 19:15:37 +00001911 }
Liz Kammer9f9fd022020-06-18 19:44:06 +00001912 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expectedPackage)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001913 }
1914}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001915
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001916func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001917 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001918 android_app {
1919 name: "foo",
1920 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001921 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001922 }
1923
1924 override_android_app {
1925 name: "bar",
1926 base: "foo",
1927 package_name: "org.dandroid.bp",
1928 }
1929
1930 android_test {
1931 name: "baz",
1932 srcs: ["b.java"],
1933 instrumentation_for: "foo",
1934 }
1935
1936 android_test {
1937 name: "qux",
1938 srcs: ["b.java"],
1939 instrumentation_for: "bar",
1940 }
1941 `)
1942
1943 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1944 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1945 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1946 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1947 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1948 }
1949
1950 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1951 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1952 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1953 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1954 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1955 }
1956}
1957
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001958func TestOverrideAndroidTest(t *testing.T) {
1959 ctx, _ := testJava(t, `
1960 android_app {
1961 name: "foo",
1962 srcs: ["a.java"],
1963 package_name: "com.android.foo",
1964 sdk_version: "current",
1965 }
1966
1967 override_android_app {
1968 name: "bar",
1969 base: "foo",
1970 package_name: "com.android.bar",
1971 }
1972
1973 android_test {
1974 name: "foo_test",
1975 srcs: ["b.java"],
1976 instrumentation_for: "foo",
1977 }
1978
1979 override_android_test {
1980 name: "bar_test",
1981 base: "foo_test",
1982 package_name: "com.android.bar.test",
1983 instrumentation_for: "bar",
1984 instrumentation_target_package: "com.android.bar",
1985 }
1986 `)
1987
1988 expectedVariants := []struct {
1989 moduleName string
1990 variantName string
1991 apkPath string
1992 overrides []string
1993 targetVariant string
1994 packageFlag string
1995 targetPackageFlag string
1996 }{
1997 {
1998 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001999 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002000 overrides: nil,
2001 targetVariant: "android_common",
2002 packageFlag: "",
2003 targetPackageFlag: "",
2004 },
2005 {
2006 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002007 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002008 overrides: []string{"foo_test"},
2009 targetVariant: "android_common_bar",
2010 packageFlag: "com.android.bar.test",
2011 targetPackageFlag: "com.android.bar",
2012 },
2013 }
2014 for _, expected := range expectedVariants {
2015 variant := ctx.ModuleForTests("foo_test", expected.variantName)
2016
2017 // Check the final apk name
2018 outputs := variant.AllOutputs()
2019 expectedApkPath := buildDir + expected.apkPath
2020 found := false
2021 for _, o := range outputs {
2022 if o == expectedApkPath {
2023 found = true
2024 break
2025 }
2026 }
2027 if !found {
2028 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
2029 }
2030
2031 // Check if the overrides field values are correctly aggregated.
2032 mod := variant.Module().(*AndroidTest)
2033 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
2034 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
2035 expected.overrides, mod.appProperties.Overrides)
2036 }
2037
2038 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
2039 javac := variant.Rule("javac")
2040 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
2041 if !strings.Contains(javac.Args["classpath"], turbine) {
2042 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
2043 }
2044
2045 // Check aapt2 flags.
2046 res := variant.Output("package-res.apk")
2047 aapt2Flags := res.Args["flags"]
2048 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammer9f9fd022020-06-18 19:44:06 +00002049 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expected.packageFlag)
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002050 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
2051 }
2052}
2053
Jaewoong Jung39982342020-01-14 10:27:18 -08002054func TestAndroidTest_FixTestConfig(t *testing.T) {
2055 ctx, _ := testJava(t, `
2056 android_app {
2057 name: "foo",
2058 srcs: ["a.java"],
2059 package_name: "com.android.foo",
2060 sdk_version: "current",
2061 }
2062
2063 android_test {
2064 name: "foo_test",
2065 srcs: ["b.java"],
2066 instrumentation_for: "foo",
2067 }
2068
2069 android_test {
2070 name: "bar_test",
2071 srcs: ["b.java"],
2072 package_name: "com.android.bar.test",
2073 instrumentation_for: "foo",
2074 }
2075
2076 override_android_test {
2077 name: "baz_test",
2078 base: "foo_test",
2079 package_name: "com.android.baz.test",
2080 }
2081 `)
2082
2083 testCases := []struct {
2084 moduleName string
2085 variantName string
2086 expectedFlags []string
2087 }{
2088 {
2089 moduleName: "foo_test",
2090 variantName: "android_common",
2091 },
2092 {
2093 moduleName: "bar_test",
2094 variantName: "android_common",
2095 expectedFlags: []string{
2096 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
2097 "--package-name com.android.bar.test",
2098 },
2099 },
2100 {
2101 moduleName: "foo_test",
2102 variantName: "android_common_baz_test",
2103 expectedFlags: []string{
2104 "--manifest " + buildDir +
2105 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
2106 "--package-name com.android.baz.test",
2107 "--test-file-name baz_test.apk",
2108 },
2109 },
2110 }
2111
2112 for _, test := range testCases {
2113 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
2114 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
2115
2116 if len(test.expectedFlags) > 0 {
2117 if params.Rule == nil {
2118 t.Errorf("test_config_fixer was expected to run, but didn't")
2119 } else {
2120 for _, flag := range test.expectedFlags {
2121 if !strings.Contains(params.RuleParams.Command, flag) {
2122 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2123 }
2124 }
2125 }
2126 } else {
2127 if params.Rule != nil {
2128 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2129 }
2130 }
2131
2132 }
2133}
2134
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002135func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002136 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002137 cc_library {
2138 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002139 sdk_version: "current",
2140 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002141 }
2142
2143 android_test {
2144 name: "stl",
2145 jni_libs: ["libjni"],
2146 compile_multilib: "both",
2147 sdk_version: "current",
2148 stl: "c++_shared",
2149 }
2150
2151 android_test {
2152 name: "system",
2153 jni_libs: ["libjni"],
2154 compile_multilib: "both",
2155 sdk_version: "current",
2156 }
2157 `)
2158
2159 testCases := []struct {
2160 name string
2161 jnis []string
2162 }{
2163 {"stl",
2164 []string{
2165 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002166 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002167 },
2168 },
2169 {"system",
2170 []string{
2171 "libjni.so",
2172 },
2173 },
2174 }
2175
2176 for _, test := range testCases {
2177 t.Run(test.name, func(t *testing.T) {
2178 app := ctx.ModuleForTests(test.name, "android_common")
2179 jniLibZip := app.Output("jnilibs.zip")
2180 var jnis []string
2181 args := strings.Fields(jniLibZip.Args["jarArgs"])
2182 for i := 0; i < len(args); i++ {
2183 if args[i] == "-f" {
2184 jnis = append(jnis, args[i+1])
2185 i += 1
2186 }
2187 }
2188 jnisJoined := strings.Join(jnis, " ")
2189 for _, jni := range test.jnis {
2190 if !strings.Contains(jnisJoined, jni) {
2191 t.Errorf("missing jni %q in %q", jni, jnis)
2192 }
2193 }
2194 })
2195 }
2196}
Colin Cross50ddcc42019-05-16 12:28:22 -07002197
2198func TestUsesLibraries(t *testing.T) {
2199 bp := `
2200 java_sdk_library {
2201 name: "foo",
2202 srcs: ["a.java"],
2203 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002204 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002205 }
2206
2207 java_sdk_library {
Paul Duffin859fe962020-05-15 10:20:31 +01002208 name: "qux",
2209 srcs: ["a.java"],
2210 api_packages: ["qux"],
2211 sdk_version: "current",
2212 }
2213
2214 java_sdk_library {
2215 name: "quuz",
2216 srcs: ["a.java"],
2217 api_packages: ["quuz"],
2218 sdk_version: "current",
2219 }
2220
2221 java_sdk_library {
Ulya Trafimovich65b03192020-12-03 16:50:22 +00002222 name: "fred",
2223 srcs: ["a.java"],
2224 api_packages: ["fred"],
2225 sdk_version: "current",
2226 }
2227
2228 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002229 name: "bar",
2230 srcs: ["a.java"],
2231 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002232 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002233 }
2234
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002235 java_sdk_library {
2236 name: "runtime-library",
2237 srcs: ["a.java"],
2238 sdk_version: "current",
2239 }
2240
2241 java_library {
2242 name: "static-runtime-helper",
2243 srcs: ["a.java"],
2244 libs: ["runtime-library"],
2245 sdk_version: "current",
2246 }
2247
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002248 // A library that has to use "provides_uses_lib", because:
2249 // - it is not an SDK library
2250 // - its library name is different from its module name
2251 java_library {
2252 name: "non-sdk-lib",
2253 provides_uses_lib: "com.non.sdk.lib",
2254 installable: true,
2255 srcs: ["a.java"],
2256 }
2257
Colin Cross50ddcc42019-05-16 12:28:22 -07002258 android_app {
2259 name: "app",
2260 srcs: ["a.java"],
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002261 libs: [
2262 "qux",
2263 "quuz.stubs"
2264 ],
Ulya Trafimovich65b03192020-12-03 16:50:22 +00002265 static_libs: [
2266 "static-runtime-helper",
2267 // statically linked component libraries should not pull their SDK libraries,
2268 // so "fred" should not be added to class loader context
2269 "fred.stubs",
2270 ],
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002271 uses_libs: [
2272 "foo",
2273 "non-sdk-lib"
2274 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002275 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002276 optional_uses_libs: [
2277 "bar",
2278 "baz",
2279 ],
2280 }
2281
2282 android_app_import {
2283 name: "prebuilt",
2284 apk: "prebuilts/apk/app.apk",
2285 certificate: "platform",
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002286 uses_libs: [
2287 "foo",
2288 "non-sdk-lib",
2289 "android.test.runner"
2290 ],
Colin Cross50ddcc42019-05-16 12:28:22 -07002291 optional_uses_libs: [
2292 "bar",
2293 "baz",
2294 ],
2295 }
2296 `
2297
Colin Cross98be1bb2019-12-13 20:41:13 -08002298 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002299 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2300
Colin Crossae8600b2020-10-29 17:09:13 -07002301 ctx := testContext(config)
Colin Cross50ddcc42019-05-16 12:28:22 -07002302
2303 run(t, ctx, config)
2304
2305 app := ctx.ModuleForTests("app", "android_common")
2306 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2307
Paul Duffin859fe962020-05-15 10:20:31 +01002308 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002309 // This should not include explicit `uses_libs`/`optional_uses_libs` entries.
2310 actualManifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2311 expectManifestFixerArgs := `--extract-native-libs=true ` +
2312 `--uses-library qux ` +
2313 `--uses-library quuz ` +
2314 `--uses-library foo ` + // TODO(b/132357300): "foo" should not be passed to manifest_fixer
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002315 `--uses-library non-sdk-lib ` + // TODO(b/132357300): "non-sdk-lib" should not be passed to manifest_fixer
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002316 `--uses-library bar ` + // TODO(b/132357300): "bar" should not be passed to manifest_fixer
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002317 `--uses-library runtime-library ` +
2318 `--uses-library com.non.sdk.lib` // TODO(b/132357300): "com.non.sdk.lib" should not be passed to manifest_fixer
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002319 if actualManifestFixerArgs != expectManifestFixerArgs {
2320 t.Errorf("unexpected manifest_fixer args:\n\texpect: %q\n\tactual: %q",
2321 expectManifestFixerArgs, actualManifestFixerArgs)
Paul Duffin859fe962020-05-15 10:20:31 +01002322 }
2323
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002324 // Test that all libraries are verified (library order matters).
2325 verifyCmd := app.Rule("verify_uses_libraries").RuleParams.Command
2326 verifyArgs := `--uses-library foo ` +
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002327 `--uses-library non-sdk-lib ` + // TODO(b/132357300): "non-sdk-lib" should not be here
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002328 `--uses-library qux ` +
2329 `--uses-library quuz ` +
2330 `--uses-library runtime-library ` +
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002331 `--uses-library com.non.sdk.lib ` +
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002332 `--optional-uses-library bar ` +
2333 `--optional-uses-library baz `
2334 if !strings.Contains(verifyCmd, verifyArgs) {
2335 t.Errorf("wanted %q in %q", verifyArgs, verifyCmd)
Colin Cross50ddcc42019-05-16 12:28:22 -07002336 }
2337
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002338 // Test that all libraries are verified for an APK (library order matters).
2339 verifyApkCmd := prebuilt.Rule("verify_uses_libraries").RuleParams.Command
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002340 // TODO(b/132357300): "non-sdk-lib" should not be here
2341 // TODO(b/132357300): "com.non.sdk.lib" should be here
2342 verifyApkReqLibs := `uses_library_names="foo non-sdk-lib android.test.runner"`
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002343 verifyApkOptLibs := `optional_uses_library_names="bar baz"`
2344 if !strings.Contains(verifyApkCmd, verifyApkReqLibs) {
2345 t.Errorf("wanted %q in %q", verifyApkReqLibs, verifyApkCmd)
Colin Cross50ddcc42019-05-16 12:28:22 -07002346 }
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002347 if !strings.Contains(verifyApkCmd, verifyApkOptLibs) {
2348 t.Errorf("wanted %q in %q", verifyApkOptLibs, verifyApkCmd)
Colin Cross50ddcc42019-05-16 12:28:22 -07002349 }
2350
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002351 // Test that all present libraries are preopted, including implicit SDK dependencies, possibly stubs
Ulya Trafimovich2eaf5c52021-02-26 12:05:11 +00002352 cmd := app.Rule("dexpreopt").RuleParams.Command
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002353 w := `--target-context-for-sdk any ` +
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002354 `PCL[/system/framework/qux.jar]#` +
Ulya Trafimovich8cbc5d22020-11-03 15:15:46 +00002355 `PCL[/system/framework/quuz.jar]#` +
2356 `PCL[/system/framework/foo.jar]#` +
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002357 `PCL[/system/framework/non-sdk-lib.jar]#` +
Ulya Trafimovich8cbc5d22020-11-03 15:15:46 +00002358 `PCL[/system/framework/bar.jar]#` +
2359 `PCL[/system/framework/runtime-library.jar]`
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002360 if !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002361 t.Errorf("wanted %q in %q", w, cmd)
2362 }
2363
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002364 // Test conditional context for target SDK version 28.
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002365 if w := `--target-context-for-sdk 28` +
2366 ` PCL[/system/framework/org.apache.http.legacy.jar] `; !strings.Contains(cmd, w) {
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002367 t.Errorf("wanted %q in %q", w, cmd)
2368 }
2369
2370 // Test conditional context for target SDK version 29.
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002371 if w := `--target-context-for-sdk 29` +
Ulya Trafimovichc9f2b942020-12-23 15:41:29 +00002372 ` PCL[/system/framework/android.hidl.manager-V1.0-java.jar]` +
2373 `#PCL[/system/framework/android.hidl.base-V1.0-java.jar] `; !strings.Contains(cmd, w) {
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002374 t.Errorf("wanted %q in %q", w, cmd)
2375 }
2376
2377 // Test conditional context for target SDK version 30.
Ulya Trafimovich46b3d5b2020-10-21 13:20:55 +01002378 // "android.test.mock" is absent because "android.test.runner" is not used.
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002379 if w := `--target-context-for-sdk 30` +
2380 ` PCL[/system/framework/android.test.base.jar] `; !strings.Contains(cmd, w) {
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002381 t.Errorf("wanted %q in %q", w, cmd)
2382 }
2383
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002384 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002385 if w := `--target-context-for-sdk any` +
2386 ` PCL[/system/framework/foo.jar]` +
Ulya Trafimovich861a8962021-02-26 14:49:07 +00002387 `#PCL[/system/framework/non-sdk-lib.jar]` +
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002388 `#PCL[/system/framework/android.test.runner.jar]` +
2389 `#PCL[/system/framework/bar.jar] `; !strings.Contains(cmd, w) {
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002390 t.Errorf("wanted %q in %q", w, cmd)
2391 }
2392
2393 // Test conditional context for target SDK version 30.
Ulya Trafimovich46b3d5b2020-10-21 13:20:55 +01002394 // "android.test.mock" is present because "android.test.runner" is used.
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002395 if w := `--target-context-for-sdk 30` +
2396 ` PCL[/system/framework/android.test.base.jar]` +
2397 `#PCL[/system/framework/android.test.mock.jar] `; !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002398 t.Errorf("wanted %q in %q", w, cmd)
2399 }
2400}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002401
2402func TestCodelessApp(t *testing.T) {
2403 testCases := []struct {
2404 name string
2405 bp string
2406 noCode bool
2407 }{
2408 {
2409 name: "normal",
2410 bp: `
2411 android_app {
2412 name: "foo",
2413 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002414 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002415 }
2416 `,
2417 noCode: false,
2418 },
2419 {
2420 name: "app without sources",
2421 bp: `
2422 android_app {
2423 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002424 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002425 }
2426 `,
2427 noCode: true,
2428 },
2429 {
2430 name: "app with libraries",
2431 bp: `
2432 android_app {
2433 name: "foo",
2434 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002435 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002436 }
2437
2438 java_library {
2439 name: "lib",
2440 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002441 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002442 }
2443 `,
2444 noCode: false,
2445 },
2446 {
2447 name: "app with sourceless libraries",
2448 bp: `
2449 android_app {
2450 name: "foo",
2451 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002452 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002453 }
2454
2455 java_library {
2456 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002457 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002458 }
2459 `,
2460 // TODO(jungjw): this should probably be true
2461 noCode: false,
2462 },
2463 }
2464
2465 for _, test := range testCases {
2466 t.Run(test.name, func(t *testing.T) {
2467 ctx := testApp(t, test.bp)
2468
2469 foo := ctx.ModuleForTests("foo", "android_common")
2470 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2471 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2472 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2473 }
2474 })
2475 }
2476}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002477
2478func TestEmbedNotice(t *testing.T) {
Colin Cross238c1f32020-06-07 16:58:18 -07002479 ctx, _ := testJavaWithFS(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002480 android_app {
2481 name: "foo",
2482 srcs: ["a.java"],
2483 static_libs: ["javalib"],
2484 jni_libs: ["libjni"],
2485 notice: "APP_NOTICE",
2486 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002487 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002488 }
2489
2490 // No embed_notice flag
2491 android_app {
2492 name: "bar",
2493 srcs: ["a.java"],
2494 jni_libs: ["libjni"],
2495 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002496 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002497 }
2498
2499 // No NOTICE files
2500 android_app {
2501 name: "baz",
2502 srcs: ["a.java"],
2503 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002504 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002505 }
2506
2507 cc_library {
2508 name: "libjni",
2509 system_shared_libs: [],
2510 stl: "none",
2511 notice: "LIB_NOTICE",
Colin Cross094cde42020-02-15 10:38:00 -08002512 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002513 }
2514
2515 java_library {
2516 name: "javalib",
2517 srcs: [
2518 ":gen",
2519 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002520 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002521 }
2522
2523 genrule {
2524 name: "gen",
2525 tools: ["gentool"],
2526 out: ["gen.java"],
2527 notice: "GENRULE_NOTICE",
2528 }
2529
2530 java_binary_host {
2531 name: "gentool",
2532 srcs: ["b.java"],
2533 notice: "TOOL_NOTICE",
2534 }
Colin Cross238c1f32020-06-07 16:58:18 -07002535 `, map[string][]byte{
2536 "APP_NOTICE": nil,
2537 "GENRULE_NOTICE": nil,
2538 "LIB_NOTICE": nil,
2539 "TOOL_NOTICE": nil,
2540 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002541
2542 // foo has NOTICE files to process, and embed_notices is true.
2543 foo := ctx.ModuleForTests("foo", "android_common")
2544 // verify merge notices rule.
2545 mergeNotices := foo.Rule("mergeNoticesRule")
2546 noticeInputs := mergeNotices.Inputs.Strings()
2547 // TOOL_NOTICE should be excluded as it's a host module.
2548 if len(mergeNotices.Inputs) != 3 {
2549 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2550 }
2551 if !inList("APP_NOTICE", noticeInputs) {
2552 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2553 }
2554 if !inList("LIB_NOTICE", noticeInputs) {
2555 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2556 }
2557 if !inList("GENRULE_NOTICE", noticeInputs) {
2558 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2559 }
2560 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2561 res := foo.Output("package-res.apk")
2562 aapt2Flags := res.Args["flags"]
2563 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2564 if !strings.Contains(aapt2Flags, e) {
2565 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2566 }
2567
2568 // bar has NOTICE files to process, but embed_notices is not set.
2569 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002570 res = bar.Output("package-res.apk")
2571 aapt2Flags = res.Args["flags"]
2572 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2573 if strings.Contains(aapt2Flags, e) {
2574 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002575 }
2576
2577 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2578 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002579 res = baz.Output("package-res.apk")
2580 aapt2Flags = res.Args["flags"]
2581 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2582 if strings.Contains(aapt2Flags, e) {
2583 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002584 }
2585}
Colin Cross53a87f52019-06-25 13:35:30 -07002586
2587func TestUncompressDex(t *testing.T) {
2588 testCases := []struct {
2589 name string
2590 bp string
2591
2592 uncompressedPlatform bool
2593 uncompressedUnbundled bool
2594 }{
2595 {
2596 name: "normal",
2597 bp: `
2598 android_app {
2599 name: "foo",
2600 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002601 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002602 }
2603 `,
2604 uncompressedPlatform: true,
2605 uncompressedUnbundled: false,
2606 },
2607 {
2608 name: "use_embedded_dex",
2609 bp: `
2610 android_app {
2611 name: "foo",
2612 use_embedded_dex: true,
2613 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002614 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002615 }
2616 `,
2617 uncompressedPlatform: true,
2618 uncompressedUnbundled: true,
2619 },
2620 {
2621 name: "privileged",
2622 bp: `
2623 android_app {
2624 name: "foo",
2625 privileged: true,
2626 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002627 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002628 }
2629 `,
2630 uncompressedPlatform: true,
2631 uncompressedUnbundled: true,
2632 },
David Srbeckye033cba2020-05-20 22:20:28 +01002633 {
2634 name: "normal_uncompress_dex_true",
2635 bp: `
2636 android_app {
2637 name: "foo",
2638 srcs: ["a.java"],
2639 sdk_version: "current",
2640 uncompress_dex: true,
2641 }
2642 `,
2643 uncompressedPlatform: true,
2644 uncompressedUnbundled: true,
2645 },
2646 {
2647 name: "normal_uncompress_dex_false",
2648 bp: `
2649 android_app {
2650 name: "foo",
2651 srcs: ["a.java"],
2652 sdk_version: "current",
2653 uncompress_dex: false,
2654 }
2655 `,
2656 uncompressedPlatform: false,
2657 uncompressedUnbundled: false,
2658 },
Colin Cross53a87f52019-06-25 13:35:30 -07002659 }
2660
2661 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2662 t.Helper()
2663
Colin Cross98be1bb2019-12-13 20:41:13 -08002664 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002665 if unbundled {
2666 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
Jeongik Cha816a23a2020-07-08 01:09:23 +09002667 config.TestProductVariables.Always_use_prebuilt_sdks = proptools.BoolPtr(true)
Colin Cross53a87f52019-06-25 13:35:30 -07002668 }
2669
Colin Crossae8600b2020-10-29 17:09:13 -07002670 ctx := testContext(config)
Colin Cross53a87f52019-06-25 13:35:30 -07002671
2672 run(t, ctx, config)
2673
2674 foo := ctx.ModuleForTests("foo", "android_common")
2675 dex := foo.Rule("r8")
2676 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2677 aligned := foo.MaybeRule("zipalign").Rule != nil
2678
2679 if uncompressedInDexJar != want {
2680 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2681 }
2682
2683 if aligned != want {
2684 t.Errorf("want aligned %v, got %v", want, aligned)
2685 }
2686 }
2687
2688 for _, tt := range testCases {
2689 t.Run(tt.name, func(t *testing.T) {
2690 t.Run("platform", func(t *testing.T) {
2691 test(t, tt.bp, tt.uncompressedPlatform, false)
2692 })
2693 t.Run("unbundled", func(t *testing.T) {
2694 test(t, tt.bp, tt.uncompressedUnbundled, true)
2695 })
2696 })
2697 }
2698}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002699
2700func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2701 if expectedValue != "" {
2702 expectedFlag := "--" + flagName + " " + expectedValue
2703 if !strings.Contains(aapt2Flags, expectedFlag) {
2704 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2705 }
2706 } else {
2707 unexpectedFlag := "--" + flagName
2708 if strings.Contains(aapt2Flags, unexpectedFlag) {
2709 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2710 }
2711 }
2712}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002713
Cole Faust9a631312020-10-22 21:05:24 +00002714func TestExportedProguardFlagFiles(t *testing.T) {
2715 ctx, _ := testJava(t, `
2716 android_app {
2717 name: "foo",
2718 sdk_version: "current",
2719 static_libs: ["lib1"],
2720 }
2721
2722 android_library {
2723 name: "lib1",
2724 sdk_version: "current",
2725 optimize: {
2726 proguard_flags_files: ["lib1proguard.cfg"],
2727 }
2728 }
2729 `)
2730
2731 m := ctx.ModuleForTests("foo", "android_common")
2732 hasLib1Proguard := false
2733 for _, s := range m.Rule("java.r8").Implicits.Strings() {
2734 if s == "lib1proguard.cfg" {
2735 hasLib1Proguard = true
2736 break
2737 }
2738 }
2739
2740 if !hasLib1Proguard {
2741 t.Errorf("App does not use library proguard config")
2742 }
2743}