blob: ec9d51b4a5e1b4fee6ee4de5926d0061e0b62f9d [file] [log] [blame]
Colin Cross3bc7ffa2017-11-22 16:19:37 -08001// Copyright 2017 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package java
16
17import (
Colin Crossd09b0b62018-04-18 11:06:47 -070018 "fmt"
Colin Crossa4f08812018-10-02 22:03:40 -070019 "path/filepath"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080020 "reflect"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070021 "regexp"
Colin Crossb69301e2017-12-01 10:48:26 -080022 "sort"
Colin Crossd09b0b62018-04-18 11:06:47 -070023 "strings"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080024 "testing"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070025
26 "github.com/google/blueprint/proptools"
27
28 "android/soong/android"
29 "android/soong/cc"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080030)
31
32var (
33 resourceFiles = []string{
34 "res/layout/layout.xml",
35 "res/values/strings.xml",
36 "res/values-en-rUS/strings.xml",
37 }
38
39 compiledResourceFiles = []string{
40 "aapt2/res/layout_layout.xml.flat",
41 "aapt2/res/values_strings.arsc.flat",
42 "aapt2/res/values-en-rUS_strings.arsc.flat",
43 }
44)
45
Colin Cross98be1bb2019-12-13 20:41:13 -080046func testAppConfig(env map[string]string, bp string, fs map[string][]byte) android.Config {
Colin Cross527012a2017-11-30 22:56:16 -080047 appFS := map[string][]byte{}
48 for k, v := range fs {
49 appFS[k] = v
Colin Cross3bc7ffa2017-11-22 16:19:37 -080050 }
51
Colin Cross527012a2017-11-30 22:56:16 -080052 for _, file := range resourceFiles {
53 appFS[file] = nil
54 }
55
Colin Cross98be1bb2019-12-13 20:41:13 -080056 return testConfig(env, bp, appFS)
Colin Cross527012a2017-11-30 22:56:16 -080057}
58
59func testApp(t *testing.T, bp string) *android.TestContext {
Colin Cross98be1bb2019-12-13 20:41:13 -080060 config := testAppConfig(nil, bp, nil)
Colin Cross527012a2017-11-30 22:56:16 -080061
Colin Cross98be1bb2019-12-13 20:41:13 -080062 ctx := testContext()
Colin Cross527012a2017-11-30 22:56:16 -080063
64 run(t, ctx, config)
65
66 return ctx
Colin Cross3bc7ffa2017-11-22 16:19:37 -080067}
68
69func TestApp(t *testing.T) {
Colin Crossa97c5d32018-03-28 14:58:31 -070070 for _, moduleType := range []string{"android_app", "android_library"} {
71 t.Run(moduleType, func(t *testing.T) {
72 ctx := testApp(t, moduleType+` {
73 name: "foo",
74 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +090075 sdk_version: "current"
Colin Crossa97c5d32018-03-28 14:58:31 -070076 }
77 `)
Colin Cross3bc7ffa2017-11-22 16:19:37 -080078
Colin Crossa97c5d32018-03-28 14:58:31 -070079 foo := ctx.ModuleForTests("foo", "android_common")
Colin Cross3bc7ffa2017-11-22 16:19:37 -080080
Colin Cross31656952018-05-24 16:11:20 -070081 var expectedLinkImplicits []string
82
83 manifestFixer := foo.Output("manifest_fixer/AndroidManifest.xml")
84 expectedLinkImplicits = append(expectedLinkImplicits, manifestFixer.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080085
Colin Crossa97c5d32018-03-28 14:58:31 -070086 frameworkRes := ctx.ModuleForTests("framework-res", "android_common")
87 expectedLinkImplicits = append(expectedLinkImplicits,
88 frameworkRes.Output("package-res.apk").Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080089
Colin Crossa97c5d32018-03-28 14:58:31 -070090 // Test the mapping from input files to compiled output file names
91 compile := foo.Output(compiledResourceFiles[0])
92 if !reflect.DeepEqual(resourceFiles, compile.Inputs.Strings()) {
93 t.Errorf("expected aapt2 compile inputs expected:\n %#v\n got:\n %#v",
94 resourceFiles, compile.Inputs.Strings())
95 }
Colin Crossb69301e2017-12-01 10:48:26 -080096
Colin Crossa97c5d32018-03-28 14:58:31 -070097 compiledResourceOutputs := compile.Outputs.Strings()
98 sort.Strings(compiledResourceOutputs)
Colin Crossb69301e2017-12-01 10:48:26 -080099
Colin Crossa97c5d32018-03-28 14:58:31 -0700100 expectedLinkImplicits = append(expectedLinkImplicits, compiledResourceOutputs...)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800101
Colin Crossa97c5d32018-03-28 14:58:31 -0700102 list := foo.Output("aapt2/res.list")
103 expectedLinkImplicits = append(expectedLinkImplicits, list.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800104
Colin Crossa97c5d32018-03-28 14:58:31 -0700105 // Check that the link rule uses
106 res := ctx.ModuleForTests("foo", "android_common").Output("package-res.apk")
107 if !reflect.DeepEqual(expectedLinkImplicits, res.Implicits.Strings()) {
108 t.Errorf("expected aapt2 link implicits expected:\n %#v\n got:\n %#v",
109 expectedLinkImplicits, res.Implicits.Strings())
110 }
111 })
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800112 }
113}
Colin Cross890ff552017-11-30 20:13:19 -0800114
Colin Crosse560c4a2019-03-19 16:03:11 -0700115func TestAppSplits(t *testing.T) {
116 ctx := testApp(t, `
117 android_app {
118 name: "foo",
119 srcs: ["a.java"],
120 package_splits: ["v4", "v7,hdpi"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900121 sdk_version: "current"
Colin Crosse560c4a2019-03-19 16:03:11 -0700122 }`)
123
124 foo := ctx.ModuleForTests("foo", "android_common")
125
126 expectedOutputs := []string{
127 filepath.Join(buildDir, ".intermediates/foo/android_common/foo.apk"),
128 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v4.apk"),
129 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v7_hdpi.apk"),
130 }
131 for _, expectedOutput := range expectedOutputs {
132 foo.Output(expectedOutput)
133 }
134
Colin Cross41955e82019-05-29 14:40:35 -0700135 outputFiles, err := foo.Module().(*AndroidApp).OutputFiles("")
136 if err != nil {
137 t.Fatal(err)
138 }
139 if g, w := outputFiles.Strings(), expectedOutputs; !reflect.DeepEqual(g, w) {
140 t.Errorf(`want OutputFiles("") = %q, got %q`, w, g)
Colin Crosse560c4a2019-03-19 16:03:11 -0700141 }
142}
143
Sasha Smundak4de27a52020-04-23 09:49:59 -0700144func TestAndroidAppSet(t *testing.T) {
145 ctx, config := testJava(t, `
146 android_app_set {
147 name: "foo",
148 set: "prebuilts/apks/app.apks",
149 prerelease: true,
150 }`)
151 module := ctx.ModuleForTests("foo", "android_common")
152 const packedSplitApks = "extracted.zip"
153 params := module.Output(packedSplitApks)
154 if params.Rule == nil {
155 t.Errorf("expected output %s is missing", packedSplitApks)
156 }
157 if s := params.Args["allow-prereleased"]; s != "true" {
158 t.Errorf("wrong allow-prereleased value: '%s', expected 'true'", s)
159 }
160 mkEntries := android.AndroidMkEntriesForTest(t, config, "", module.Module())[0]
161 actualMaster := mkEntries.EntryMap["LOCAL_APK_SET_MASTER_FILE"]
162 expectedMaster := []string{"foo.apk"}
163 if !reflect.DeepEqual(actualMaster, expectedMaster) {
164 t.Errorf("Unexpected LOCAL_APK_SET_MASTER_FILE value: '%s', expected: '%s',",
165 actualMaster, expectedMaster)
166 }
167}
168
169func TestAndroidAppSet_Variants(t *testing.T) {
170 bp := `
171 android_app_set {
172 name: "foo",
173 set: "prebuilts/apks/app.apks",
174 }`
175 testCases := []struct {
176 name string
177 deviceArch *string
178 deviceSecondaryArch *string
179 aaptPrebuiltDPI []string
180 sdkVersion int
181 expected map[string]string
182 }{
183 {
184 name: "One",
185 deviceArch: proptools.StringPtr("x86"),
186 aaptPrebuiltDPI: []string{"ldpi", "xxhdpi"},
187 sdkVersion: 29,
188 expected: map[string]string{
189 "abis": "X86",
190 "allow-prereleased": "false",
191 "screen-densities": "LDPI,XXHDPI",
192 "sdk-version": "29",
193 "stem": "foo",
194 },
195 },
196 {
197 name: "Two",
198 deviceArch: proptools.StringPtr("x86_64"),
199 deviceSecondaryArch: proptools.StringPtr("x86"),
200 aaptPrebuiltDPI: nil,
201 sdkVersion: 30,
202 expected: map[string]string{
203 "abis": "X86_64,X86",
204 "allow-prereleased": "false",
205 "screen-densities": "all",
206 "sdk-version": "30",
207 "stem": "foo",
208 },
209 },
210 }
211
212 for _, test := range testCases {
213 config := testAppConfig(nil, bp, nil)
214 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
215 config.TestProductVariables.Platform_sdk_version = &test.sdkVersion
216 config.TestProductVariables.DeviceArch = test.deviceArch
217 config.TestProductVariables.DeviceSecondaryArch = test.deviceSecondaryArch
218 ctx := testContext()
219 run(t, ctx, config)
220 module := ctx.ModuleForTests("foo", "android_common")
221 const packedSplitApks = "extracted.zip"
222 params := module.Output(packedSplitApks)
223 for k, v := range test.expected {
224 if actual := params.Args[k]; actual != v {
225 t.Errorf("%s: bad build arg value for '%s': '%s', expected '%s'",
226 test.name, k, actual, v)
227 }
228 }
229 }
230}
231
Jeongik Cha538c0d02019-07-11 15:54:27 +0900232func TestPlatformAPIs(t *testing.T) {
233 testJava(t, `
234 android_app {
235 name: "foo",
236 srcs: ["a.java"],
237 platform_apis: true,
238 }
239 `)
240
241 testJava(t, `
242 android_app {
243 name: "foo",
244 srcs: ["a.java"],
245 sdk_version: "current",
246 }
247 `)
248
249 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
250 android_app {
251 name: "bar",
252 srcs: ["b.java"],
253 }
254 `)
255
256 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
257 android_app {
258 name: "bar",
259 srcs: ["b.java"],
260 sdk_version: "system_current",
261 platform_apis: true,
262 }
263 `)
264}
265
Jeongik Chae403e9e2019-12-07 00:16:24 +0900266func TestAndroidAppLinkType(t *testing.T) {
267 testJava(t, `
268 android_app {
269 name: "foo",
270 srcs: ["a.java"],
271 libs: ["bar"],
272 static_libs: ["baz"],
273 platform_apis: true,
274 }
275
276 java_library {
277 name: "bar",
278 sdk_version: "current",
279 srcs: ["b.java"],
280 }
281
282 android_library {
283 name: "baz",
284 sdk_version: "system_current",
285 srcs: ["c.java"],
286 }
287 `)
288
289 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
290 android_app {
291 name: "foo",
292 srcs: ["a.java"],
293 libs: ["bar"],
294 sdk_version: "current",
295 static_libs: ["baz"],
296 }
297
298 java_library {
299 name: "bar",
300 sdk_version: "current",
301 srcs: ["b.java"],
302 }
303
304 android_library {
305 name: "baz",
306 sdk_version: "system_current",
307 srcs: ["c.java"],
308 }
309 `)
310
311 testJava(t, `
312 android_app {
313 name: "foo",
314 srcs: ["a.java"],
315 libs: ["bar"],
316 sdk_version: "system_current",
317 static_libs: ["baz"],
318 }
319
320 java_library {
321 name: "bar",
322 sdk_version: "current",
323 srcs: ["b.java"],
324 }
325
326 android_library {
327 name: "baz",
328 sdk_version: "system_current",
329 srcs: ["c.java"],
330 }
331 `)
332
333 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
334 android_app {
335 name: "foo",
336 srcs: ["a.java"],
337 libs: ["bar"],
338 sdk_version: "system_current",
339 static_libs: ["baz"],
340 }
341
342 java_library {
343 name: "bar",
344 sdk_version: "current",
345 srcs: ["b.java"],
346 }
347
348 android_library {
349 name: "baz",
350 srcs: ["c.java"],
351 }
352 `)
353}
354
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100355func TestUpdatableApps(t *testing.T) {
356 testCases := []struct {
357 name string
358 bp string
359 expectedError string
360 }{
361 {
362 name: "Stable public SDK",
363 bp: `android_app {
364 name: "foo",
365 srcs: ["a.java"],
366 sdk_version: "29",
Artur Satayev11962102020-04-16 13:43:02 +0100367 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100368 updatable: true,
369 }`,
370 },
371 {
372 name: "Stable system SDK",
373 bp: `android_app {
374 name: "foo",
375 srcs: ["a.java"],
376 sdk_version: "system_29",
Artur Satayev11962102020-04-16 13:43:02 +0100377 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100378 updatable: true,
379 }`,
380 },
381 {
382 name: "Current public SDK",
383 bp: `android_app {
384 name: "foo",
385 srcs: ["a.java"],
386 sdk_version: "current",
Artur Satayev11962102020-04-16 13:43:02 +0100387 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100388 updatable: true,
389 }`,
390 },
391 {
392 name: "Current system SDK",
393 bp: `android_app {
394 name: "foo",
395 srcs: ["a.java"],
396 sdk_version: "system_current",
Artur Satayev11962102020-04-16 13:43:02 +0100397 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100398 updatable: true,
399 }`,
400 },
401 {
402 name: "Current module SDK",
403 bp: `android_app {
404 name: "foo",
405 srcs: ["a.java"],
406 sdk_version: "module_current",
Artur Satayev11962102020-04-16 13:43:02 +0100407 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100408 updatable: true,
409 }`,
410 },
411 {
412 name: "Current core SDK",
413 bp: `android_app {
414 name: "foo",
415 srcs: ["a.java"],
416 sdk_version: "core_current",
Artur Satayev11962102020-04-16 13:43:02 +0100417 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100418 updatable: true,
419 }`,
420 },
421 {
422 name: "No Platform APIs",
423 bp: `android_app {
424 name: "foo",
425 srcs: ["a.java"],
426 platform_apis: true,
Artur Satayev11962102020-04-16 13:43:02 +0100427 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100428 updatable: true,
429 }`,
430 expectedError: "Updatable apps must use stable SDKs",
431 },
432 {
433 name: "No Core Platform APIs",
434 bp: `android_app {
435 name: "foo",
436 srcs: ["a.java"],
437 sdk_version: "core_platform",
Artur Satayev11962102020-04-16 13:43:02 +0100438 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100439 updatable: true,
440 }`,
441 expectedError: "Updatable apps must use stable SDKs",
442 },
443 {
444 name: "No unspecified APIs",
445 bp: `android_app {
446 name: "foo",
447 srcs: ["a.java"],
448 updatable: true,
Artur Satayev11962102020-04-16 13:43:02 +0100449 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100450 }`,
451 expectedError: "Updatable apps must use stable SDK",
452 },
Artur Satayev11962102020-04-16 13:43:02 +0100453 {
454 name: "Must specify min_sdk_version",
455 bp: `android_app {
456 name: "app_without_min_sdk_version",
457 srcs: ["a.java"],
458 sdk_version: "29",
459 updatable: true,
460 }`,
461 expectedError: "updatable apps must set min_sdk_version.",
462 },
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100463 }
464
465 for _, test := range testCases {
466 t.Run(test.name, func(t *testing.T) {
467 if test.expectedError == "" {
468 testJava(t, test.bp)
469 } else {
470 testJavaError(t, test.expectedError, test.bp)
471 }
472 })
473 }
474}
475
Colin Cross0ddae7f2019-02-07 15:30:01 -0800476func TestResourceDirs(t *testing.T) {
477 testCases := []struct {
478 name string
479 prop string
480 resources []string
481 }{
482 {
483 name: "no resource_dirs",
484 prop: "",
485 resources: []string{"res/res/values/strings.xml"},
486 },
487 {
488 name: "resource_dirs",
489 prop: `resource_dirs: ["res"]`,
490 resources: []string{"res/res/values/strings.xml"},
491 },
492 {
493 name: "empty resource_dirs",
494 prop: `resource_dirs: []`,
495 resources: nil,
496 },
497 }
498
499 fs := map[string][]byte{
500 "res/res/values/strings.xml": nil,
501 }
502
503 bp := `
504 android_app {
505 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900506 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800507 %s
508 }
509 `
510
511 for _, testCase := range testCases {
512 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800513 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
514 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800515 run(t, ctx, config)
516
517 module := ctx.ModuleForTests("foo", "android_common")
518 resourceList := module.MaybeOutput("aapt2/res.list")
519
520 var resources []string
521 if resourceList.Rule != nil {
522 for _, compiledResource := range resourceList.Inputs.Strings() {
523 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
524 }
525 }
526
527 if !reflect.DeepEqual(resources, testCase.resources) {
528 t.Errorf("expected resource files %q, got %q",
529 testCase.resources, resources)
530 }
531 })
532 }
533}
534
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800535func TestLibraryAssets(t *testing.T) {
536 bp := `
537 android_app {
538 name: "foo",
539 sdk_version: "current",
540 static_libs: ["lib1", "lib2", "lib3"],
541 }
542
543 android_library {
544 name: "lib1",
545 sdk_version: "current",
546 asset_dirs: ["assets_a"],
547 }
548
549 android_library {
550 name: "lib2",
551 sdk_version: "current",
552 }
553
554 android_library {
555 name: "lib3",
556 sdk_version: "current",
557 static_libs: ["lib4"],
558 }
559
560 android_library {
561 name: "lib4",
562 sdk_version: "current",
563 asset_dirs: ["assets_b"],
564 }
565 `
566
567 testCases := []struct {
568 name string
569 assetFlag string
570 assetPackages []string
571 }{
572 {
573 name: "foo",
574 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
575 assetPackages: []string{
576 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
577 buildDir + "/.intermediates/lib1/android_common/assets.zip",
578 buildDir + "/.intermediates/lib3/android_common/assets.zip",
579 },
580 },
581 {
582 name: "lib1",
583 assetFlag: "-A assets_a",
584 },
585 {
586 name: "lib2",
587 },
588 {
589 name: "lib3",
590 assetPackages: []string{
591 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
592 buildDir + "/.intermediates/lib4/android_common/assets.zip",
593 },
594 },
595 {
596 name: "lib4",
597 assetFlag: "-A assets_b",
598 },
599 }
600 ctx := testApp(t, bp)
601
602 for _, test := range testCases {
603 t.Run(test.name, func(t *testing.T) {
604 m := ctx.ModuleForTests(test.name, "android_common")
605
606 // Check asset flag in aapt2 link flags
607 var aapt2link android.TestingBuildParams
608 if len(test.assetPackages) > 0 {
609 aapt2link = m.Output("aapt2/package-res.apk")
610 } else {
611 aapt2link = m.Output("package-res.apk")
612 }
613 aapt2Flags := aapt2link.Args["flags"]
614 if test.assetFlag != "" {
615 if !strings.Contains(aapt2Flags, test.assetFlag) {
616 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
617 }
618 } else {
619 if strings.Contains(aapt2Flags, " -A ") {
620 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
621 }
622 }
623
624 // Check asset merge rule.
625 if len(test.assetPackages) > 0 {
626 mergeAssets := m.Output("package-res.apk")
627 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
628 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
629 mergeAssets.Inputs.Strings(), test.assetPackages)
630 }
631 }
632 })
633 }
634}
635
Colin Crossbec85302019-02-13 13:15:46 -0800636func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800637 testCases := []struct {
638 name string
639 enforceRROTargets []string
640 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800641 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800642 overlayFiles map[string][]string
643 rroDirs map[string][]string
644 }{
645 {
646 name: "no RRO",
647 enforceRROTargets: nil,
648 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800649 resourceFiles: map[string][]string{
650 "foo": nil,
651 "bar": {"bar/res/res/values/strings.xml"},
652 "lib": nil,
653 "lib2": {"lib2/res/res/values/strings.xml"},
654 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800655 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800656 "foo": {
657 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800658 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000659 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800660 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800661 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
662 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000663 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800664 },
Colin Crossbec85302019-02-13 13:15:46 -0800665 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800666 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
667 "device/vendor/blah/overlay/bar/res/values/strings.xml",
668 },
Colin Crossbec85302019-02-13 13:15:46 -0800669 "lib": {
670 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
671 "lib/res/res/values/strings.xml",
672 "device/vendor/blah/overlay/lib/res/values/strings.xml",
673 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800674 },
675 rroDirs: map[string][]string{
676 "foo": nil,
677 "bar": nil,
678 },
679 },
680 {
681 name: "enforce RRO on foo",
682 enforceRROTargets: []string{"foo"},
683 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800684 resourceFiles: map[string][]string{
685 "foo": nil,
686 "bar": {"bar/res/res/values/strings.xml"},
687 "lib": nil,
688 "lib2": {"lib2/res/res/values/strings.xml"},
689 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800690 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800691 "foo": {
692 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800693 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000694 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800695 "foo/res/res/values/strings.xml",
696 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
697 },
Colin Crossbec85302019-02-13 13:15:46 -0800698 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800699 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
700 "device/vendor/blah/overlay/bar/res/values/strings.xml",
701 },
Colin Crossbec85302019-02-13 13:15:46 -0800702 "lib": {
703 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
704 "lib/res/res/values/strings.xml",
705 "device/vendor/blah/overlay/lib/res/values/strings.xml",
706 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800707 },
Colin Crossc1c37552019-01-31 11:42:41 -0800708
Colin Cross5c4791c2019-02-01 11:44:44 -0800709 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800710 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000711 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800712 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
713 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000714 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800715 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800716 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800717 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800718 },
719 },
720 {
721 name: "enforce RRO on all",
722 enforceRROTargets: []string{"*"},
723 enforceRROExcludedOverlays: []string{
724 // Excluding specific apps/res directories also allowed.
725 "device/vendor/blah/static_overlay/foo",
726 "device/vendor/blah/static_overlay/bar/res",
727 },
Colin Crossbec85302019-02-13 13:15:46 -0800728 resourceFiles: map[string][]string{
729 "foo": nil,
730 "bar": {"bar/res/res/values/strings.xml"},
731 "lib": nil,
732 "lib2": {"lib2/res/res/values/strings.xml"},
733 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800734 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800735 "foo": {
736 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800737 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000738 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800739 "foo/res/res/values/strings.xml",
740 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
741 },
Colin Crossbec85302019-02-13 13:15:46 -0800742 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
743 "lib": {
744 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
745 "lib/res/res/values/strings.xml",
746 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800747 },
748 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800749 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000750 "device:device/vendor/blah/overlay/foo/res",
751 "product:product/vendor/blah/overlay/foo/res",
752 // Lib dep comes after the direct deps
753 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800754 },
Anton Hansson53c88442019-03-18 15:53:16 +0000755 "bar": {"device:device/vendor/blah/overlay/bar/res"},
756 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800757 },
758 },
759 }
760
Anton Hansson53c88442019-03-18 15:53:16 +0000761 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800762 "device/vendor/blah/overlay",
763 "device/vendor/blah/overlay2",
764 "device/vendor/blah/static_overlay",
765 }
766
Anton Hansson53c88442019-03-18 15:53:16 +0000767 productResourceOverlays := []string{
768 "product/vendor/blah/overlay",
769 }
770
Colin Cross890ff552017-11-30 20:13:19 -0800771 fs := map[string][]byte{
772 "foo/res/res/values/strings.xml": nil,
773 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800774 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800775 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800776 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
777 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800778 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800779 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
780 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
781 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000782 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800783 }
784
785 bp := `
786 android_app {
787 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900788 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800789 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000790 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800791 }
792
793 android_app {
794 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900795 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800796 resource_dirs: ["bar/res"],
797 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800798
799 android_library {
800 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900801 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800802 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800803 static_libs: ["lib2"],
804 }
805
806 android_library {
807 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900808 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800809 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800810 }
Anton Hansson53c88442019-03-18 15:53:16 +0000811
812 // This library has the same resources as lib (should not lead to dupe RROs)
813 android_library {
814 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900815 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000816 resource_dirs: ["lib/res"]
817 }
Colin Cross890ff552017-11-30 20:13:19 -0800818 `
819
Colin Cross5c4791c2019-02-01 11:44:44 -0800820 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800821 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800822 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000823 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
824 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800825 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800826 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800827 }
828 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800829 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800830 }
831
Colin Cross98be1bb2019-12-13 20:41:13 -0800832 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800833 run(t, ctx, config)
834
Colin Crossbec85302019-02-13 13:15:46 -0800835 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
836 for _, o := range list {
837 res := module.MaybeOutput(o)
838 if res.Rule != nil {
839 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
840 // verify the inputs to the .arsc.flat rule.
841 files = append(files, res.Inputs.Strings()...)
842 } else {
843 // Otherwise, verify the full path to the output of the other module
844 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000845 }
Colin Cross890ff552017-11-30 20:13:19 -0800846 }
Colin Crossbec85302019-02-13 13:15:46 -0800847 return files
Colin Cross890ff552017-11-30 20:13:19 -0800848 }
849
Colin Crossbec85302019-02-13 13:15:46 -0800850 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
851 module := ctx.ModuleForTests(moduleName, "android_common")
852 resourceList := module.MaybeOutput("aapt2/res.list")
853 if resourceList.Rule != nil {
854 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000855 }
Colin Crossbec85302019-02-13 13:15:46 -0800856 overlayList := module.MaybeOutput("aapt2/overlay.list")
857 if overlayList.Rule != nil {
858 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
859 }
860
Anton Hansson53c88442019-03-18 15:53:16 +0000861 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
862 var prefix string
863 if d.overlayType == device {
864 prefix = "device:"
865 } else if d.overlayType == product {
866 prefix = "product:"
867 } else {
868 t.Fatalf("Unexpected overlayType %d", d.overlayType)
869 }
870 rroDirs = append(rroDirs, prefix+d.path.String())
871 }
Colin Crossbec85302019-02-13 13:15:46 -0800872
873 return resourceFiles, overlayFiles, rroDirs
874 }
875
876 modules := []string{"foo", "bar", "lib", "lib2"}
877 for _, module := range modules {
878 resourceFiles, overlayFiles, rroDirs := getResources(module)
879
880 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
881 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
882 module, testCase.resourceFiles[module], resourceFiles)
883 }
884 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
885 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
886 module, testCase.overlayFiles[module], overlayFiles)
887 }
888 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +0000889 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -0800890 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +0000891 }
Colin Cross890ff552017-11-30 20:13:19 -0800892 }
Colin Cross890ff552017-11-30 20:13:19 -0800893 })
894 }
895}
Colin Crossd09b0b62018-04-18 11:06:47 -0700896
897func TestAppSdkVersion(t *testing.T) {
898 testCases := []struct {
899 name string
900 sdkVersion string
901 platformSdkInt int
902 platformSdkCodename string
903 platformSdkFinal bool
904 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +0900905 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -0700906 }{
907 {
908 name: "current final SDK",
909 sdkVersion: "current",
910 platformSdkInt: 27,
911 platformSdkCodename: "REL",
912 platformSdkFinal: true,
913 expectedMinSdkVersion: "27",
914 },
915 {
916 name: "current non-final SDK",
917 sdkVersion: "current",
918 platformSdkInt: 27,
919 platformSdkCodename: "OMR1",
920 platformSdkFinal: false,
921 expectedMinSdkVersion: "OMR1",
922 },
923 {
924 name: "default final SDK",
925 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900926 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700927 platformSdkInt: 27,
928 platformSdkCodename: "REL",
929 platformSdkFinal: true,
930 expectedMinSdkVersion: "27",
931 },
932 {
933 name: "default non-final SDK",
934 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900935 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700936 platformSdkInt: 27,
937 platformSdkCodename: "OMR1",
938 platformSdkFinal: false,
939 expectedMinSdkVersion: "OMR1",
940 },
941 {
942 name: "14",
943 sdkVersion: "14",
944 expectedMinSdkVersion: "14",
945 },
946 }
947
948 for _, moduleType := range []string{"android_app", "android_library"} {
949 for _, test := range testCases {
950 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +0900951 platformApiProp := ""
952 if test.platformApis {
953 platformApiProp = "platform_apis: true,"
954 }
Colin Crossd09b0b62018-04-18 11:06:47 -0700955 bp := fmt.Sprintf(`%s {
956 name: "foo",
957 srcs: ["a.java"],
958 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900959 %s
960 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -0700961
Colin Cross98be1bb2019-12-13 20:41:13 -0800962 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -0700963 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
964 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
965 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
966
Colin Cross98be1bb2019-12-13 20:41:13 -0800967 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -0700968
969 run(t, ctx, config)
970
971 foo := ctx.ModuleForTests("foo", "android_common")
972 link := foo.Output("package-res.apk")
973 linkFlags := strings.Split(link.Args["flags"], " ")
974 min := android.IndexList("--min-sdk-version", linkFlags)
975 target := android.IndexList("--target-sdk-version", linkFlags)
976
977 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
978 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
979 }
980
981 gotMinSdkVersion := linkFlags[min+1]
982 gotTargetSdkVersion := linkFlags[target+1]
983
984 if gotMinSdkVersion != test.expectedMinSdkVersion {
985 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
986 test.expectedMinSdkVersion, gotMinSdkVersion)
987 }
988
989 if gotTargetSdkVersion != test.expectedMinSdkVersion {
990 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
991 test.expectedMinSdkVersion, gotTargetSdkVersion)
992 }
993 })
994 }
995 }
996}
Colin Crossa4f08812018-10-02 22:03:40 -0700997
Paul Duffin50c217c2019-06-12 13:25:22 +0100998func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700999 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001000 cc_library {
1001 name: "libjni",
1002 system_shared_libs: [],
Colin Cross01fd7cc2020-02-19 16:54:04 -08001003 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001004 stl: "none",
1005 }
1006
1007 android_test {
1008 name: "test",
1009 sdk_version: "core_platform",
1010 jni_libs: ["libjni"],
1011 }
1012
1013 android_test {
1014 name: "test_first",
1015 sdk_version: "core_platform",
1016 compile_multilib: "first",
1017 jni_libs: ["libjni"],
1018 }
1019
1020 android_test {
1021 name: "test_both",
1022 sdk_version: "core_platform",
1023 compile_multilib: "both",
1024 jni_libs: ["libjni"],
1025 }
1026
1027 android_test {
1028 name: "test_32",
1029 sdk_version: "core_platform",
1030 compile_multilib: "32",
1031 jni_libs: ["libjni"],
1032 }
1033
1034 android_test {
1035 name: "test_64",
1036 sdk_version: "core_platform",
1037 compile_multilib: "64",
1038 jni_libs: ["libjni"],
1039 }
1040 `)
1041
1042 testCases := []struct {
1043 name string
1044 abis []string
1045 }{
1046 {"test", []string{"arm64-v8a"}},
1047 {"test_first", []string{"arm64-v8a"}},
1048 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1049 {"test_32", []string{"armeabi-v7a"}},
1050 {"test_64", []string{"arm64-v8a"}},
1051 }
1052
1053 for _, test := range testCases {
1054 t.Run(test.name, func(t *testing.T) {
1055 app := ctx.ModuleForTests(test.name, "android_common")
1056 jniLibZip := app.Output("jnilibs.zip")
1057 var abis []string
1058 args := strings.Fields(jniLibZip.Args["jarArgs"])
1059 for i := 0; i < len(args); i++ {
1060 if args[i] == "-P" {
1061 abis = append(abis, filepath.Base(args[i+1]))
1062 i++
1063 }
1064 }
1065 if !reflect.DeepEqual(abis, test.abis) {
1066 t.Errorf("want abis %v, got %v", test.abis, abis)
1067 }
1068 })
1069 }
1070}
1071
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001072func TestAppSdkVersionByPartition(t *testing.T) {
1073 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1074 android_app {
1075 name: "foo",
1076 srcs: ["a.java"],
1077 vendor: true,
1078 platform_apis: true,
1079 }
1080 `)
1081
1082 testJava(t, `
1083 android_app {
1084 name: "bar",
1085 srcs: ["b.java"],
1086 platform_apis: true,
1087 }
1088 `)
1089
1090 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001091 bp := `
1092 android_app {
1093 name: "foo",
1094 srcs: ["a.java"],
1095 product_specific: true,
1096 platform_apis: true,
1097 }
1098 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001099
1100 config := testAppConfig(nil, bp, nil)
1101 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001102 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001103 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 +09001104 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001105 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001106 }
1107 }
1108}
1109
Paul Duffin50c217c2019-06-12 13:25:22 +01001110func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001111 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001112 cc_library {
1113 name: "libjni",
1114 system_shared_libs: [],
1115 stl: "none",
Colin Cross1c93c292020-02-15 10:38:00 -08001116 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001117 }
1118
1119 android_app {
1120 name: "app",
1121 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001122 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001123 }
1124
1125 android_app {
1126 name: "app_noembed",
1127 jni_libs: ["libjni"],
1128 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001129 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001130 }
1131
1132 android_app {
1133 name: "app_embed",
1134 jni_libs: ["libjni"],
1135 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001136 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001137 }
1138
1139 android_test {
1140 name: "test",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001141 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001142 jni_libs: ["libjni"],
1143 }
1144
1145 android_test {
1146 name: "test_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001147 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001148 jni_libs: ["libjni"],
1149 use_embedded_native_libs: false,
1150 }
1151
1152 android_test_helper_app {
1153 name: "test_helper",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001154 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001155 jni_libs: ["libjni"],
1156 }
1157
1158 android_test_helper_app {
1159 name: "test_helper_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001160 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001161 jni_libs: ["libjni"],
1162 use_embedded_native_libs: false,
1163 }
1164 `)
1165
1166 testCases := []struct {
1167 name string
1168 packaged bool
1169 compressed bool
1170 }{
1171 {"app", false, false},
1172 {"app_noembed", false, false},
1173 {"app_embed", true, false},
1174 {"test", true, false},
1175 {"test_noembed", true, true},
1176 {"test_helper", true, false},
1177 {"test_helper_noembed", true, true},
1178 }
1179
1180 for _, test := range testCases {
1181 t.Run(test.name, func(t *testing.T) {
1182 app := ctx.ModuleForTests(test.name, "android_common")
1183 jniLibZip := app.MaybeOutput("jnilibs.zip")
1184 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1185 t.Errorf("expected jni packaged %v, got %v", w, g)
1186 }
1187
1188 if jniLibZip.Rule != nil {
1189 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1190 t.Errorf("expected jni compressed %v, got %v", w, g)
1191 }
Colin Cross01fd7cc2020-02-19 16:54:04 -08001192
1193 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1194 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1195 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001196 }
1197 })
1198 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001199}
1200
Colin Cross1dd9c442020-05-08 11:20:24 -07001201func TestJNISDK(t *testing.T) {
1202 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1203 cc_library {
1204 name: "libjni",
1205 system_shared_libs: [],
1206 stl: "none",
1207 sdk_version: "current",
1208 }
1209
1210 android_test {
1211 name: "app_platform",
1212 jni_libs: ["libjni"],
1213 platform_apis: true,
1214 }
1215
1216 android_test {
1217 name: "app_sdk",
1218 jni_libs: ["libjni"],
1219 sdk_version: "current",
1220 }
1221
1222 android_test {
1223 name: "app_force_platform",
1224 jni_libs: ["libjni"],
1225 sdk_version: "current",
1226 jni_uses_platform_apis: true,
1227 }
1228
1229 android_test {
1230 name: "app_force_sdk",
1231 jni_libs: ["libjni"],
1232 platform_apis: true,
1233 jni_uses_sdk_apis: true,
1234 }
Colin Crosseb032962020-05-13 11:05:02 -07001235
1236 cc_library {
1237 name: "libvendorjni",
1238 system_shared_libs: [],
1239 stl: "none",
1240 vendor: true,
1241 }
1242
1243 android_test {
1244 name: "app_vendor",
1245 jni_libs: ["libvendorjni"],
1246 sdk_version: "current",
1247 vendor: true,
1248 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001249 `)
1250
1251 testCases := []struct {
Colin Crosseb032962020-05-13 11:05:02 -07001252 name string
1253 sdkJNI bool
1254 vendorJNI bool
Colin Cross1dd9c442020-05-08 11:20:24 -07001255 }{
Colin Crosseb032962020-05-13 11:05:02 -07001256 {name: "app_platform"},
1257 {name: "app_sdk", sdkJNI: true},
1258 {name: "app_force_platform"},
1259 {name: "app_force_sdk", sdkJNI: true},
1260 {name: "app_vendor", vendorJNI: true},
Colin Cross1dd9c442020-05-08 11:20:24 -07001261 }
1262
Colin Crosseb032962020-05-13 11:05:02 -07001263 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1264 Output("libjni.so").Output.String()
1265 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1266 Output("libjni.so").Output.String()
1267 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1268 Output("libvendorjni.so").Output.String()
1269
Colin Cross1dd9c442020-05-08 11:20:24 -07001270 for _, test := range testCases {
1271 t.Run(test.name, func(t *testing.T) {
1272 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross1dd9c442020-05-08 11:20:24 -07001273
1274 jniLibZip := app.MaybeOutput("jnilibs.zip")
1275 if len(jniLibZip.Implicits) != 1 {
1276 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1277 }
1278 gotJNI := jniLibZip.Implicits[0].String()
1279
1280 if test.sdkJNI {
1281 if gotJNI != sdkJNI {
1282 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1283 }
Colin Crosseb032962020-05-13 11:05:02 -07001284 } else if test.vendorJNI {
1285 if gotJNI != vendorJNI {
1286 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1287 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001288 } else {
1289 if gotJNI != platformJNI {
1290 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1291 }
1292 }
1293 })
1294 }
1295
1296 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1297 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1298 android_test {
1299 name: "app_platform",
1300 platform_apis: true,
1301 jni_uses_platform_apis: true,
1302 }
1303 `)
1304 })
1305
1306 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1307 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1308 android_test {
1309 name: "app_sdk",
1310 sdk_version: "current",
1311 jni_uses_sdk_apis: true,
1312 }
1313 `)
1314 })
1315
1316}
1317
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001318func TestCertificates(t *testing.T) {
1319 testCases := []struct {
1320 name string
1321 bp string
1322 certificateOverride string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001323 expectedLineage string
1324 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001325 }{
1326 {
1327 name: "default",
1328 bp: `
1329 android_app {
1330 name: "foo",
1331 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001332 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001333 }
1334 `,
1335 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001336 expectedLineage: "",
1337 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001338 },
1339 {
1340 name: "module certificate property",
1341 bp: `
1342 android_app {
1343 name: "foo",
1344 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001345 certificate: ":new_certificate",
1346 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001347 }
1348
1349 android_app_certificate {
1350 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001351 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001352 }
1353 `,
1354 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001355 expectedLineage: "",
1356 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001357 },
1358 {
1359 name: "path certificate property",
1360 bp: `
1361 android_app {
1362 name: "foo",
1363 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001364 certificate: "expiredkey",
1365 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001366 }
1367 `,
1368 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001369 expectedLineage: "",
1370 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001371 },
1372 {
1373 name: "certificate overrides",
1374 bp: `
1375 android_app {
1376 name: "foo",
1377 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001378 certificate: "expiredkey",
1379 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001380 }
1381
1382 android_app_certificate {
1383 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001384 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001385 }
1386 `,
1387 certificateOverride: "foo:new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001388 expectedLineage: "",
1389 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1390 },
1391 {
1392 name: "certificate lineage",
1393 bp: `
1394 android_app {
1395 name: "foo",
1396 srcs: ["a.java"],
1397 certificate: ":new_certificate",
1398 lineage: "lineage.bin",
1399 sdk_version: "current",
1400 }
1401
1402 android_app_certificate {
1403 name: "new_certificate",
1404 certificate: "cert/new_cert",
1405 }
1406 `,
1407 certificateOverride: "",
1408 expectedLineage: "--lineage lineage.bin",
1409 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001410 },
1411 }
1412
1413 for _, test := range testCases {
1414 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001415 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001416 if test.certificateOverride != "" {
1417 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1418 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001419 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001420
1421 run(t, ctx, config)
1422 foo := ctx.ModuleForTests("foo", "android_common")
1423
1424 signapk := foo.Output("foo.apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001425 signCertificateFlags := signapk.Args["certificates"]
1426 if test.expectedCertificate != signCertificateFlags {
1427 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1428 }
1429
1430 signFlags := signapk.Args["flags"]
1431 if test.expectedLineage != signFlags {
1432 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001433 }
1434 })
1435 }
1436}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001437
Songchun Fan688de9a2020-03-24 20:32:24 -07001438func TestRequestV4SigningFlag(t *testing.T) {
1439 testCases := []struct {
1440 name string
1441 bp string
1442 expected string
1443 }{
1444 {
1445 name: "default",
1446 bp: `
1447 android_app {
1448 name: "foo",
1449 srcs: ["a.java"],
1450 sdk_version: "current",
1451 }
1452 `,
1453 expected: "",
1454 },
1455 {
1456 name: "default",
1457 bp: `
1458 android_app {
1459 name: "foo",
1460 srcs: ["a.java"],
1461 sdk_version: "current",
1462 v4_signature: false,
1463 }
1464 `,
1465 expected: "",
1466 },
1467 {
1468 name: "module certificate property",
1469 bp: `
1470 android_app {
1471 name: "foo",
1472 srcs: ["a.java"],
1473 sdk_version: "current",
1474 v4_signature: true,
1475 }
1476 `,
1477 expected: "--enable-v4",
1478 },
1479 }
1480
1481 for _, test := range testCases {
1482 t.Run(test.name, func(t *testing.T) {
1483 config := testAppConfig(nil, test.bp, nil)
1484 ctx := testContext()
1485
1486 run(t, ctx, config)
1487 foo := ctx.ModuleForTests("foo", "android_common")
1488
1489 signapk := foo.Output("foo.apk")
1490 signFlags := signapk.Args["flags"]
1491 if test.expected != signFlags {
1492 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1493 }
1494 })
1495 }
1496}
1497
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001498func TestPackageNameOverride(t *testing.T) {
1499 testCases := []struct {
1500 name string
1501 bp string
1502 packageNameOverride string
1503 expected []string
1504 }{
1505 {
1506 name: "default",
1507 bp: `
1508 android_app {
1509 name: "foo",
1510 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001511 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001512 }
1513 `,
1514 packageNameOverride: "",
1515 expected: []string{
1516 buildDir + "/.intermediates/foo/android_common/foo.apk",
1517 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1518 },
1519 },
1520 {
1521 name: "overridden",
1522 bp: `
1523 android_app {
1524 name: "foo",
1525 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001526 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001527 }
1528 `,
1529 packageNameOverride: "foo:bar",
1530 expected: []string{
1531 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001532 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001533 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1534 },
1535 },
1536 }
1537
1538 for _, test := range testCases {
1539 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001540 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001541 if test.packageNameOverride != "" {
1542 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1543 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001544 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001545
1546 run(t, ctx, config)
1547 foo := ctx.ModuleForTests("foo", "android_common")
1548
1549 outputs := foo.AllOutputs()
1550 outputMap := make(map[string]bool)
1551 for _, o := range outputs {
1552 outputMap[o] = true
1553 }
1554 for _, e := range test.expected {
1555 if _, exist := outputMap[e]; !exist {
1556 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1557 }
1558 }
1559 })
1560 }
1561}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001562
1563func TestInstrumentationTargetOverridden(t *testing.T) {
1564 bp := `
1565 android_app {
1566 name: "foo",
1567 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001568 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001569 }
1570
1571 android_test {
1572 name: "bar",
1573 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001574 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001575 }
1576 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001577 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001578 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001579 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001580
1581 run(t, ctx, config)
1582
1583 bar := ctx.ModuleForTests("bar", "android_common")
1584 res := bar.Output("package-res.apk")
1585 aapt2Flags := res.Args["flags"]
1586 e := "--rename-instrumentation-target-package org.dandroid.bp"
1587 if !strings.Contains(aapt2Flags, e) {
1588 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1589 }
1590}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001591
1592func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001593 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001594 android_app {
1595 name: "foo",
1596 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001597 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001598 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001599 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001600 }
1601
1602 override_android_app {
1603 name: "bar",
1604 base: "foo",
1605 certificate: ":new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001606 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001607 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001608 }
1609
1610 android_app_certificate {
1611 name: "new_certificate",
1612 certificate: "cert/new_cert",
1613 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001614
1615 override_android_app {
1616 name: "baz",
1617 base: "foo",
1618 package_name: "org.dandroid.bp",
1619 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001620 `)
1621
1622 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001623 moduleName string
1624 variantName string
1625 apkName string
1626 apkPath string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001627 certFlag string
1628 lineageFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001629 overrides []string
1630 aaptFlag string
1631 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001632 }{
1633 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001634 moduleName: "foo",
1635 variantName: "android_common",
1636 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001637 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1638 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001639 overrides: []string{"qux"},
1640 aaptFlag: "",
1641 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001642 },
1643 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001644 moduleName: "bar",
1645 variantName: "android_common_bar",
1646 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001647 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1648 lineageFlag: "--lineage lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001649 overrides: []string{"qux", "foo"},
1650 aaptFlag: "",
1651 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001652 },
1653 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001654 moduleName: "baz",
1655 variantName: "android_common_baz",
1656 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001657 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1658 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001659 overrides: []string{"qux", "foo"},
1660 aaptFlag: "--rename-manifest-package org.dandroid.bp",
1661 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001662 },
1663 }
1664 for _, expected := range expectedVariants {
1665 variant := ctx.ModuleForTests("foo", expected.variantName)
1666
1667 // Check the final apk name
1668 outputs := variant.AllOutputs()
1669 expectedApkPath := buildDir + expected.apkPath
1670 found := false
1671 for _, o := range outputs {
1672 if o == expectedApkPath {
1673 found = true
1674 break
1675 }
1676 }
1677 if !found {
1678 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1679 }
1680
1681 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001682 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001683 certFlag := signapk.Args["certificates"]
1684 if expected.certFlag != certFlag {
1685 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1686 }
1687
1688 // Check the lineage flags
1689 lineageFlag := signapk.Args["flags"]
1690 if expected.lineageFlag != lineageFlag {
1691 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001692 }
1693
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001694 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001695 mod := variant.Module().(*AndroidApp)
1696 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1697 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1698 expected.overrides, mod.appProperties.Overrides)
1699 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001700
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001701 // Test Overridable property: Logging_parent
1702 logging_parent := mod.aapt.LoggingParent
1703 if expected.logging_parent != logging_parent {
1704 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1705 expected.logging_parent, logging_parent)
1706 }
1707
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001708 // Check the package renaming flag, if exists.
1709 res := variant.Output("package-res.apk")
1710 aapt2Flags := res.Args["flags"]
1711 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1712 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1713 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001714 }
1715}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001716
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001717func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001718 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001719 android_app {
1720 name: "foo",
1721 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001722 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001723 }
1724
1725 override_android_app {
1726 name: "bar",
1727 base: "foo",
1728 package_name: "org.dandroid.bp",
1729 }
1730
1731 android_test {
1732 name: "baz",
1733 srcs: ["b.java"],
1734 instrumentation_for: "foo",
1735 }
1736
1737 android_test {
1738 name: "qux",
1739 srcs: ["b.java"],
1740 instrumentation_for: "bar",
1741 }
1742 `)
1743
1744 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1745 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1746 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1747 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1748 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1749 }
1750
1751 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1752 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1753 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1754 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1755 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1756 }
1757}
1758
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001759func TestOverrideAndroidTest(t *testing.T) {
1760 ctx, _ := testJava(t, `
1761 android_app {
1762 name: "foo",
1763 srcs: ["a.java"],
1764 package_name: "com.android.foo",
1765 sdk_version: "current",
1766 }
1767
1768 override_android_app {
1769 name: "bar",
1770 base: "foo",
1771 package_name: "com.android.bar",
1772 }
1773
1774 android_test {
1775 name: "foo_test",
1776 srcs: ["b.java"],
1777 instrumentation_for: "foo",
1778 }
1779
1780 override_android_test {
1781 name: "bar_test",
1782 base: "foo_test",
1783 package_name: "com.android.bar.test",
1784 instrumentation_for: "bar",
1785 instrumentation_target_package: "com.android.bar",
1786 }
1787 `)
1788
1789 expectedVariants := []struct {
1790 moduleName string
1791 variantName string
1792 apkPath string
1793 overrides []string
1794 targetVariant string
1795 packageFlag string
1796 targetPackageFlag string
1797 }{
1798 {
1799 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001800 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001801 overrides: nil,
1802 targetVariant: "android_common",
1803 packageFlag: "",
1804 targetPackageFlag: "",
1805 },
1806 {
1807 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001808 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001809 overrides: []string{"foo_test"},
1810 targetVariant: "android_common_bar",
1811 packageFlag: "com.android.bar.test",
1812 targetPackageFlag: "com.android.bar",
1813 },
1814 }
1815 for _, expected := range expectedVariants {
1816 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1817
1818 // Check the final apk name
1819 outputs := variant.AllOutputs()
1820 expectedApkPath := buildDir + expected.apkPath
1821 found := false
1822 for _, o := range outputs {
1823 if o == expectedApkPath {
1824 found = true
1825 break
1826 }
1827 }
1828 if !found {
1829 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1830 }
1831
1832 // Check if the overrides field values are correctly aggregated.
1833 mod := variant.Module().(*AndroidTest)
1834 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1835 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1836 expected.overrides, mod.appProperties.Overrides)
1837 }
1838
1839 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1840 javac := variant.Rule("javac")
1841 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1842 if !strings.Contains(javac.Args["classpath"], turbine) {
1843 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1844 }
1845
1846 // Check aapt2 flags.
1847 res := variant.Output("package-res.apk")
1848 aapt2Flags := res.Args["flags"]
1849 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1850 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1851 }
1852}
1853
Jaewoong Jung39982342020-01-14 10:27:18 -08001854func TestAndroidTest_FixTestConfig(t *testing.T) {
1855 ctx, _ := testJava(t, `
1856 android_app {
1857 name: "foo",
1858 srcs: ["a.java"],
1859 package_name: "com.android.foo",
1860 sdk_version: "current",
1861 }
1862
1863 android_test {
1864 name: "foo_test",
1865 srcs: ["b.java"],
1866 instrumentation_for: "foo",
1867 }
1868
1869 android_test {
1870 name: "bar_test",
1871 srcs: ["b.java"],
1872 package_name: "com.android.bar.test",
1873 instrumentation_for: "foo",
1874 }
1875
1876 override_android_test {
1877 name: "baz_test",
1878 base: "foo_test",
1879 package_name: "com.android.baz.test",
1880 }
1881 `)
1882
1883 testCases := []struct {
1884 moduleName string
1885 variantName string
1886 expectedFlags []string
1887 }{
1888 {
1889 moduleName: "foo_test",
1890 variantName: "android_common",
1891 },
1892 {
1893 moduleName: "bar_test",
1894 variantName: "android_common",
1895 expectedFlags: []string{
1896 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
1897 "--package-name com.android.bar.test",
1898 },
1899 },
1900 {
1901 moduleName: "foo_test",
1902 variantName: "android_common_baz_test",
1903 expectedFlags: []string{
1904 "--manifest " + buildDir +
1905 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
1906 "--package-name com.android.baz.test",
1907 "--test-file-name baz_test.apk",
1908 },
1909 },
1910 }
1911
1912 for _, test := range testCases {
1913 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
1914 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
1915
1916 if len(test.expectedFlags) > 0 {
1917 if params.Rule == nil {
1918 t.Errorf("test_config_fixer was expected to run, but didn't")
1919 } else {
1920 for _, flag := range test.expectedFlags {
1921 if !strings.Contains(params.RuleParams.Command, flag) {
1922 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
1923 }
1924 }
1925 }
1926 } else {
1927 if params.Rule != nil {
1928 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
1929 }
1930 }
1931
1932 }
1933}
1934
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001935func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001936 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001937 android_app_import {
1938 name: "foo",
1939 apk: "prebuilts/apk/app.apk",
1940 certificate: "platform",
1941 dex_preopt: {
1942 enabled: true,
1943 },
1944 }
1945 `)
1946
1947 variant := ctx.ModuleForTests("foo", "android_common")
1948
1949 // Check dexpreopt outputs.
1950 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1951 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1952 t.Errorf("can't find dexpreopt outputs")
1953 }
1954
1955 // Check cert signing flag.
1956 signedApk := variant.Output("signed/foo.apk")
1957 signingFlag := signedApk.Args["certificates"]
1958 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
1959 if expected != signingFlag {
1960 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1961 }
1962}
1963
1964func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001965 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001966 android_app_import {
1967 name: "foo",
1968 apk: "prebuilts/apk/app.apk",
1969 certificate: "platform",
1970 dex_preopt: {
1971 enabled: false,
1972 },
1973 }
1974 `)
1975
1976 variant := ctx.ModuleForTests("foo", "android_common")
1977
1978 // Check dexpreopt outputs. They shouldn't exist.
1979 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
1980 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
1981 t.Errorf("dexpreopt shouldn't have run.")
1982 }
1983}
1984
1985func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001986 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001987 android_app_import {
1988 name: "foo",
1989 apk: "prebuilts/apk/app.apk",
1990 presigned: true,
1991 dex_preopt: {
1992 enabled: true,
1993 },
1994 }
1995 `)
1996
1997 variant := ctx.ModuleForTests("foo", "android_common")
1998
1999 // Check dexpreopt outputs.
2000 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2001 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2002 t.Errorf("can't find dexpreopt outputs")
2003 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002004 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002005 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2006 t.Errorf("signing rule shouldn't be included.")
2007 }
2008 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2009 t.Errorf("can't find aligning rule")
2010 }
2011}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002012
Liz Kammer2bc57f62020-05-13 15:49:21 -07002013func TestAndroidAppImport_SigningLineage(t *testing.T) {
2014 ctx, _ := testJava(t, `
2015 android_app_import {
2016 name: "foo",
2017 apk: "prebuilts/apk/app.apk",
2018 certificate: "platform",
2019 lineage: "lineage.bin",
2020 }
2021 `)
2022
2023 variant := ctx.ModuleForTests("foo", "android_common")
2024
2025 // Check cert signing lineage flag.
2026 signedApk := variant.Output("signed/foo.apk")
2027 signingFlag := signedApk.Args["flags"]
2028 expected := "--lineage lineage.bin"
2029 if expected != signingFlag {
2030 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2031 }
2032}
2033
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002034func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2035 ctx, _ := testJava(t, `
2036 android_app_import {
2037 name: "foo",
2038 apk: "prebuilts/apk/app.apk",
2039 default_dev_cert: true,
2040 dex_preopt: {
2041 enabled: true,
2042 },
2043 }
2044 `)
2045
2046 variant := ctx.ModuleForTests("foo", "android_common")
2047
2048 // Check dexpreopt outputs.
2049 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2050 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2051 t.Errorf("can't find dexpreopt outputs")
2052 }
2053
2054 // Check cert signing flag.
2055 signedApk := variant.Output("signed/foo.apk")
2056 signingFlag := signedApk.Args["certificates"]
2057 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2058 if expected != signingFlag {
2059 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2060 }
2061}
2062
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002063func TestAndroidAppImport_DpiVariants(t *testing.T) {
2064 bp := `
2065 android_app_import {
2066 name: "foo",
2067 apk: "prebuilts/apk/app.apk",
2068 dpi_variants: {
2069 xhdpi: {
2070 apk: "prebuilts/apk/app_xhdpi.apk",
2071 },
2072 xxhdpi: {
2073 apk: "prebuilts/apk/app_xxhdpi.apk",
2074 },
2075 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002076 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002077 dex_preopt: {
2078 enabled: true,
2079 },
2080 }
2081 `
2082 testCases := []struct {
2083 name string
2084 aaptPreferredConfig *string
2085 aaptPrebuiltDPI []string
2086 expected string
2087 }{
2088 {
2089 name: "no preferred",
2090 aaptPreferredConfig: nil,
2091 aaptPrebuiltDPI: []string{},
2092 expected: "prebuilts/apk/app.apk",
2093 },
2094 {
2095 name: "AAPTPreferredConfig matches",
2096 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002097 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002098 expected: "prebuilts/apk/app_xhdpi.apk",
2099 },
2100 {
2101 name: "AAPTPrebuiltDPI matches",
2102 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2103 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2104 expected: "prebuilts/apk/app_xxhdpi.apk",
2105 },
2106 {
2107 name: "non-first AAPTPrebuiltDPI matches",
2108 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2109 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2110 expected: "prebuilts/apk/app_xhdpi.apk",
2111 },
2112 {
2113 name: "no matches",
2114 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2115 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2116 expected: "prebuilts/apk/app.apk",
2117 },
2118 }
2119
2120 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2121 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002122 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002123 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2124 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002125 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002126
2127 run(t, ctx, config)
2128
2129 variant := ctx.ModuleForTests("foo", "android_common")
2130 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2131 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2132 if len(matches) != 2 {
2133 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2134 }
2135 if test.expected != matches[1] {
2136 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2137 }
2138 }
2139}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002140
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002141func TestAndroidAppImport_Filename(t *testing.T) {
2142 ctx, config := testJava(t, `
2143 android_app_import {
2144 name: "foo",
2145 apk: "prebuilts/apk/app.apk",
2146 presigned: true,
2147 }
2148
2149 android_app_import {
2150 name: "bar",
2151 apk: "prebuilts/apk/app.apk",
2152 presigned: true,
2153 filename: "bar_sample.apk"
2154 }
2155 `)
2156
2157 testCases := []struct {
2158 name string
2159 expected string
2160 }{
2161 {
2162 name: "foo",
2163 expected: "foo.apk",
2164 },
2165 {
2166 name: "bar",
2167 expected: "bar_sample.apk",
2168 },
2169 }
2170
2171 for _, test := range testCases {
2172 variant := ctx.ModuleForTests(test.name, "android_common")
2173 if variant.MaybeOutput(test.expected).Rule == nil {
2174 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2175 }
2176
2177 a := variant.Module().(*AndroidAppImport)
2178 expectedValues := []string{test.expected}
2179 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002180 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002181 if !reflect.DeepEqual(actualValues, expectedValues) {
2182 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2183 actualValues, expectedValues)
2184 }
2185 }
2186}
2187
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002188func TestAndroidAppImport_ArchVariants(t *testing.T) {
2189 // The test config's target arch is ARM64.
2190 testCases := []struct {
2191 name string
2192 bp string
2193 expected string
2194 }{
2195 {
2196 name: "matching arch",
2197 bp: `
2198 android_app_import {
2199 name: "foo",
2200 apk: "prebuilts/apk/app.apk",
2201 arch: {
2202 arm64: {
2203 apk: "prebuilts/apk/app_arm64.apk",
2204 },
2205 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002206 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002207 dex_preopt: {
2208 enabled: true,
2209 },
2210 }
2211 `,
2212 expected: "prebuilts/apk/app_arm64.apk",
2213 },
2214 {
2215 name: "no matching arch",
2216 bp: `
2217 android_app_import {
2218 name: "foo",
2219 apk: "prebuilts/apk/app.apk",
2220 arch: {
2221 arm: {
2222 apk: "prebuilts/apk/app_arm.apk",
2223 },
2224 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002225 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002226 dex_preopt: {
2227 enabled: true,
2228 },
2229 }
2230 `,
2231 expected: "prebuilts/apk/app.apk",
2232 },
2233 }
2234
2235 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2236 for _, test := range testCases {
2237 ctx, _ := testJava(t, test.bp)
2238
2239 variant := ctx.ModuleForTests("foo", "android_common")
2240 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2241 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2242 if len(matches) != 2 {
2243 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2244 }
2245 if test.expected != matches[1] {
2246 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2247 }
2248 }
2249}
2250
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002251func TestAndroidTestImport(t *testing.T) {
2252 ctx, config := testJava(t, `
2253 android_test_import {
2254 name: "foo",
2255 apk: "prebuilts/apk/app.apk",
2256 presigned: true,
2257 data: [
2258 "testdata/data",
2259 ],
2260 }
2261 `)
2262
2263 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2264
2265 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002266 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002267 expected := []string{"tests"}
2268 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2269 if !reflect.DeepEqual(expected, actual) {
2270 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2271 }
2272 expected = []string{"testdata/data:testdata/data"}
2273 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2274 if !reflect.DeepEqual(expected, actual) {
2275 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2276 }
2277}
2278
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002279func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2280 ctx, _ := testJava(t, `
2281 android_test_import {
2282 name: "foo",
2283 apk: "prebuilts/apk/app.apk",
2284 certificate: "cert/new_cert",
2285 data: [
2286 "testdata/data",
2287 ],
2288 }
2289
2290 android_test_import {
2291 name: "foo_presigned",
2292 apk: "prebuilts/apk/app.apk",
2293 presigned: true,
2294 data: [
2295 "testdata/data",
2296 ],
2297 }
2298 `)
2299
2300 variant := ctx.ModuleForTests("foo", "android_common")
2301 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2302 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2303 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2304 }
2305
2306 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2307 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2308 if jniRule != android.Cp.String() {
2309 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2310 }
2311}
2312
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002313func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002314 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002315 cc_library {
2316 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002317 sdk_version: "current",
2318 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002319 }
2320
2321 android_test {
2322 name: "stl",
2323 jni_libs: ["libjni"],
2324 compile_multilib: "both",
2325 sdk_version: "current",
2326 stl: "c++_shared",
2327 }
2328
2329 android_test {
2330 name: "system",
2331 jni_libs: ["libjni"],
2332 compile_multilib: "both",
2333 sdk_version: "current",
2334 }
2335 `)
2336
2337 testCases := []struct {
2338 name string
2339 jnis []string
2340 }{
2341 {"stl",
2342 []string{
2343 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002344 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002345 },
2346 },
2347 {"system",
2348 []string{
2349 "libjni.so",
2350 },
2351 },
2352 }
2353
2354 for _, test := range testCases {
2355 t.Run(test.name, func(t *testing.T) {
2356 app := ctx.ModuleForTests(test.name, "android_common")
2357 jniLibZip := app.Output("jnilibs.zip")
2358 var jnis []string
2359 args := strings.Fields(jniLibZip.Args["jarArgs"])
2360 for i := 0; i < len(args); i++ {
2361 if args[i] == "-f" {
2362 jnis = append(jnis, args[i+1])
2363 i += 1
2364 }
2365 }
2366 jnisJoined := strings.Join(jnis, " ")
2367 for _, jni := range test.jnis {
2368 if !strings.Contains(jnisJoined, jni) {
2369 t.Errorf("missing jni %q in %q", jni, jnis)
2370 }
2371 }
2372 })
2373 }
2374}
Colin Cross50ddcc42019-05-16 12:28:22 -07002375
2376func TestUsesLibraries(t *testing.T) {
2377 bp := `
2378 java_sdk_library {
2379 name: "foo",
2380 srcs: ["a.java"],
2381 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002382 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002383 }
2384
2385 java_sdk_library {
2386 name: "bar",
2387 srcs: ["a.java"],
2388 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002389 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002390 }
2391
2392 android_app {
2393 name: "app",
2394 srcs: ["a.java"],
2395 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002396 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002397 optional_uses_libs: [
2398 "bar",
2399 "baz",
2400 ],
2401 }
2402
2403 android_app_import {
2404 name: "prebuilt",
2405 apk: "prebuilts/apk/app.apk",
2406 certificate: "platform",
2407 uses_libs: ["foo"],
2408 optional_uses_libs: [
2409 "bar",
2410 "baz",
2411 ],
2412 }
2413 `
2414
Colin Cross98be1bb2019-12-13 20:41:13 -08002415 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002416 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2417
Colin Cross98be1bb2019-12-13 20:41:13 -08002418 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002419
2420 run(t, ctx, config)
2421
2422 app := ctx.ModuleForTests("app", "android_common")
2423 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2424
2425 // Test that all libraries are verified
2426 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2427 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2428 t.Errorf("wanted %q in %q", w, cmd)
2429 }
2430
2431 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2432 t.Errorf("wanted %q in %q", w, cmd)
2433 }
2434
2435 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2436
2437 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2438 t.Errorf("wanted %q in %q", w, cmd)
2439 }
2440
2441 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2442 t.Errorf("wanted %q in %q", w, cmd)
2443 }
2444
2445 // Test that only present libraries are preopted
2446 cmd = app.Rule("dexpreopt").RuleParams.Command
2447
2448 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2449 t.Errorf("wanted %q in %q", w, cmd)
2450 }
2451
2452 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2453
2454 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2455 t.Errorf("wanted %q in %q", w, cmd)
2456 }
2457}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002458
2459func TestCodelessApp(t *testing.T) {
2460 testCases := []struct {
2461 name string
2462 bp string
2463 noCode bool
2464 }{
2465 {
2466 name: "normal",
2467 bp: `
2468 android_app {
2469 name: "foo",
2470 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002471 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002472 }
2473 `,
2474 noCode: false,
2475 },
2476 {
2477 name: "app without sources",
2478 bp: `
2479 android_app {
2480 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002481 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002482 }
2483 `,
2484 noCode: true,
2485 },
2486 {
2487 name: "app with libraries",
2488 bp: `
2489 android_app {
2490 name: "foo",
2491 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002492 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002493 }
2494
2495 java_library {
2496 name: "lib",
2497 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002498 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002499 }
2500 `,
2501 noCode: false,
2502 },
2503 {
2504 name: "app with sourceless libraries",
2505 bp: `
2506 android_app {
2507 name: "foo",
2508 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002509 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002510 }
2511
2512 java_library {
2513 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002514 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002515 }
2516 `,
2517 // TODO(jungjw): this should probably be true
2518 noCode: false,
2519 },
2520 }
2521
2522 for _, test := range testCases {
2523 t.Run(test.name, func(t *testing.T) {
2524 ctx := testApp(t, test.bp)
2525
2526 foo := ctx.ModuleForTests("foo", "android_common")
2527 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2528 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2529 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2530 }
2531 })
2532 }
2533}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002534
2535func TestEmbedNotice(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002536 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002537 android_app {
2538 name: "foo",
2539 srcs: ["a.java"],
2540 static_libs: ["javalib"],
2541 jni_libs: ["libjni"],
2542 notice: "APP_NOTICE",
2543 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002544 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002545 }
2546
2547 // No embed_notice flag
2548 android_app {
2549 name: "bar",
2550 srcs: ["a.java"],
2551 jni_libs: ["libjni"],
2552 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002553 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002554 }
2555
2556 // No NOTICE files
2557 android_app {
2558 name: "baz",
2559 srcs: ["a.java"],
2560 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002561 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002562 }
2563
2564 cc_library {
2565 name: "libjni",
2566 system_shared_libs: [],
2567 stl: "none",
2568 notice: "LIB_NOTICE",
Colin Cross1c93c292020-02-15 10:38:00 -08002569 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002570 }
2571
2572 java_library {
2573 name: "javalib",
2574 srcs: [
2575 ":gen",
2576 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002577 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002578 }
2579
2580 genrule {
2581 name: "gen",
2582 tools: ["gentool"],
2583 out: ["gen.java"],
2584 notice: "GENRULE_NOTICE",
2585 }
2586
2587 java_binary_host {
2588 name: "gentool",
2589 srcs: ["b.java"],
2590 notice: "TOOL_NOTICE",
2591 }
2592 `)
2593
2594 // foo has NOTICE files to process, and embed_notices is true.
2595 foo := ctx.ModuleForTests("foo", "android_common")
2596 // verify merge notices rule.
2597 mergeNotices := foo.Rule("mergeNoticesRule")
2598 noticeInputs := mergeNotices.Inputs.Strings()
2599 // TOOL_NOTICE should be excluded as it's a host module.
2600 if len(mergeNotices.Inputs) != 3 {
2601 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2602 }
2603 if !inList("APP_NOTICE", noticeInputs) {
2604 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2605 }
2606 if !inList("LIB_NOTICE", noticeInputs) {
2607 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2608 }
2609 if !inList("GENRULE_NOTICE", noticeInputs) {
2610 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2611 }
2612 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2613 res := foo.Output("package-res.apk")
2614 aapt2Flags := res.Args["flags"]
2615 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2616 if !strings.Contains(aapt2Flags, e) {
2617 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2618 }
2619
2620 // bar has NOTICE files to process, but embed_notices is not set.
2621 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002622 res = bar.Output("package-res.apk")
2623 aapt2Flags = res.Args["flags"]
2624 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2625 if strings.Contains(aapt2Flags, e) {
2626 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002627 }
2628
2629 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2630 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002631 res = baz.Output("package-res.apk")
2632 aapt2Flags = res.Args["flags"]
2633 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2634 if strings.Contains(aapt2Flags, e) {
2635 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002636 }
2637}
Colin Cross53a87f52019-06-25 13:35:30 -07002638
2639func TestUncompressDex(t *testing.T) {
2640 testCases := []struct {
2641 name string
2642 bp string
2643
2644 uncompressedPlatform bool
2645 uncompressedUnbundled bool
2646 }{
2647 {
2648 name: "normal",
2649 bp: `
2650 android_app {
2651 name: "foo",
2652 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002653 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002654 }
2655 `,
2656 uncompressedPlatform: true,
2657 uncompressedUnbundled: false,
2658 },
2659 {
2660 name: "use_embedded_dex",
2661 bp: `
2662 android_app {
2663 name: "foo",
2664 use_embedded_dex: true,
2665 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002666 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002667 }
2668 `,
2669 uncompressedPlatform: true,
2670 uncompressedUnbundled: true,
2671 },
2672 {
2673 name: "privileged",
2674 bp: `
2675 android_app {
2676 name: "foo",
2677 privileged: true,
2678 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002679 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002680 }
2681 `,
2682 uncompressedPlatform: true,
2683 uncompressedUnbundled: true,
2684 },
David Srbecky98c71222020-05-20 22:20:28 +01002685 {
2686 name: "normal_uncompress_dex_true",
2687 bp: `
2688 android_app {
2689 name: "foo",
2690 srcs: ["a.java"],
2691 sdk_version: "current",
2692 uncompress_dex: true,
2693 }
2694 `,
2695 uncompressedPlatform: true,
2696 uncompressedUnbundled: true,
2697 },
2698 {
2699 name: "normal_uncompress_dex_false",
2700 bp: `
2701 android_app {
2702 name: "foo",
2703 srcs: ["a.java"],
2704 sdk_version: "current",
2705 uncompress_dex: false,
2706 }
2707 `,
2708 uncompressedPlatform: false,
2709 uncompressedUnbundled: false,
2710 },
Colin Cross53a87f52019-06-25 13:35:30 -07002711 }
2712
2713 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2714 t.Helper()
2715
Colin Cross98be1bb2019-12-13 20:41:13 -08002716 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002717 if unbundled {
2718 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2719 }
2720
Colin Cross98be1bb2019-12-13 20:41:13 -08002721 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002722
2723 run(t, ctx, config)
2724
2725 foo := ctx.ModuleForTests("foo", "android_common")
2726 dex := foo.Rule("r8")
2727 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2728 aligned := foo.MaybeRule("zipalign").Rule != nil
2729
2730 if uncompressedInDexJar != want {
2731 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2732 }
2733
2734 if aligned != want {
2735 t.Errorf("want aligned %v, got %v", want, aligned)
2736 }
2737 }
2738
2739 for _, tt := range testCases {
2740 t.Run(tt.name, func(t *testing.T) {
2741 t.Run("platform", func(t *testing.T) {
2742 test(t, tt.bp, tt.uncompressedPlatform, false)
2743 })
2744 t.Run("unbundled", func(t *testing.T) {
2745 test(t, tt.bp, tt.uncompressedUnbundled, true)
2746 })
2747 })
2748 }
2749}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002750
2751func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2752 if expectedValue != "" {
2753 expectedFlag := "--" + flagName + " " + expectedValue
2754 if !strings.Contains(aapt2Flags, expectedFlag) {
2755 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2756 }
2757 } else {
2758 unexpectedFlag := "--" + flagName
2759 if strings.Contains(aapt2Flags, unexpectedFlag) {
2760 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2761 }
2762 }
2763}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002764
2765func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungca095d72020-04-09 16:15:30 -07002766 fs := map[string][]byte{
2767 "baz/res/res/values/strings.xml": nil,
2768 "bar/res/res/values/strings.xml": nil,
2769 }
2770 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002771 runtime_resource_overlay {
2772 name: "foo",
2773 certificate: "platform",
2774 product_specific: true,
Jaewoong Jungca095d72020-04-09 16:15:30 -07002775 static_libs: ["bar"],
2776 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002777 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002778 }
2779
2780 runtime_resource_overlay {
2781 name: "foo_themed",
2782 certificate: "platform",
2783 product_specific: true,
2784 theme: "faza",
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002785 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002786 }
Jaewoong Jungca095d72020-04-09 16:15:30 -07002787
2788 android_library {
2789 name: "bar",
2790 resource_dirs: ["bar/res"],
2791 }
2792
2793 android_app {
2794 name: "baz",
2795 sdk_version: "current",
2796 resource_dirs: ["baz/res"],
2797 }
2798 `
2799 config := testAppConfig(nil, bp, fs)
2800 ctx := testContext()
2801 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002802
2803 m := ctx.ModuleForTests("foo", "android_common")
2804
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002805 // Check AAPT2 link flags.
2806 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2807 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2808 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2809 if len(absentFlags) > 0 {
2810 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2811 }
2812
Jaewoong Jungca095d72020-04-09 16:15:30 -07002813 // Check overlay.list output for static_libs dependency.
2814 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
2815 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
2816 if !inList(staticLibPackage, overlayList) {
2817 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
2818 }
2819
2820 // Check AAPT2 link flags for resource_libs dependency.
2821 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
2822 if !strings.Contains(aapt2Flags, resourceLibFlag) {
2823 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
2824 }
2825
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002826 // Check cert signing flag.
2827 signedApk := m.Output("signed/foo.apk")
2828 signingFlag := signedApk.Args["certificates"]
2829 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2830 if expected != signingFlag {
2831 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2832 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002833 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2834 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002835 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
2836 if !reflect.DeepEqual(path, expectedPath) {
2837 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
2838 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002839
2840 // Check device location.
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002841 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002842 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002843 if !reflect.DeepEqual(path, expectedPath) {
2844 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2845 }
2846
2847 // A themed module has a different device location
2848 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002849 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2850 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002851 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
2852 if !reflect.DeepEqual(path, expectedPath) {
2853 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2854 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002855
2856 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
2857 expectedOverrides := []string{"foo"}
2858 if !reflect.DeepEqual(overrides, expectedOverrides) {
2859 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
2860 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002861}
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07002862
2863func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
2864 ctx, config := testJava(t, `
2865 java_defaults {
2866 name: "rro_defaults",
2867 theme: "default_theme",
2868 product_specific: true,
2869 aaptflags: ["--keep-raw-values"],
2870 }
2871
2872 runtime_resource_overlay {
2873 name: "foo_with_defaults",
2874 defaults: ["rro_defaults"],
2875 }
2876
2877 runtime_resource_overlay {
2878 name: "foo_barebones",
2879 }
2880 `)
2881
2882 //
2883 // RRO module with defaults
2884 //
2885 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
2886
2887 // Check AAPT2 link flags.
2888 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
2889 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2890 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
2891 if len(absentFlags) > 0 {
2892 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2893 }
2894
2895 // Check device location.
2896 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2897 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
2898 if !reflect.DeepEqual(path, expectedPath) {
2899 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
2900 }
2901
2902 //
2903 // RRO module without defaults
2904 //
2905 m = ctx.ModuleForTests("foo_barebones", "android_common")
2906
2907 // Check AAPT2 link flags.
2908 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
2909 unexpectedFlags := "--keep-raw-values"
2910 if inList(unexpectedFlags, aapt2Flags) {
2911 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
2912 }
2913
2914 // Check device location.
2915 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2916 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
2917 if !reflect.DeepEqual(path, expectedPath) {
2918 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2919 }
2920}
Roshan Piusb8307962020-04-27 09:42:27 -07002921
2922func TestOverrideRuntimeResourceOverlay(t *testing.T) {
2923 ctx, _ := testJava(t, `
2924 runtime_resource_overlay {
2925 name: "foo_overlay",
2926 certificate: "platform",
2927 product_specific: true,
2928 sdk_version: "current",
2929 }
2930
2931 override_runtime_resource_overlay {
2932 name: "bar_overlay",
2933 base: "foo_overlay",
2934 package_name: "com.android.bar.overlay",
2935 target_package_name: "com.android.bar",
2936 }
2937 `)
2938
2939 expectedVariants := []struct {
2940 moduleName string
2941 variantName string
2942 apkPath string
2943 overrides []string
2944 targetVariant string
2945 packageFlag string
2946 targetPackageFlag string
2947 }{
2948 {
2949 variantName: "android_common",
2950 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
2951 overrides: nil,
2952 targetVariant: "android_common",
2953 packageFlag: "",
2954 targetPackageFlag: "",
2955 },
2956 {
2957 variantName: "android_common_bar_overlay",
2958 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
2959 overrides: []string{"foo_overlay"},
2960 targetVariant: "android_common_bar",
2961 packageFlag: "com.android.bar.overlay",
2962 targetPackageFlag: "com.android.bar",
2963 },
2964 }
2965 for _, expected := range expectedVariants {
2966 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
2967
2968 // Check the final apk name
2969 outputs := variant.AllOutputs()
2970 expectedApkPath := buildDir + expected.apkPath
2971 found := false
2972 for _, o := range outputs {
2973 if o == expectedApkPath {
2974 found = true
2975 break
2976 }
2977 }
2978 if !found {
2979 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
2980 }
2981
2982 // Check if the overrides field values are correctly aggregated.
2983 mod := variant.Module().(*RuntimeResourceOverlay)
2984 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
2985 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
2986 expected.overrides, mod.properties.Overrides)
2987 }
2988
2989 // Check aapt2 flags.
2990 res := variant.Output("package-res.apk")
2991 aapt2Flags := res.Args["flags"]
2992 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
2993 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
2994 }
2995}