blob: beb29a78ccc9287e063af59114a54a004ca58a5b [file] [log] [blame]
Colin Cross3bc7ffa2017-11-22 16:19:37 -08001// Copyright 2017 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package java
16
17import (
Colin Crossd09b0b62018-04-18 11:06:47 -070018 "fmt"
Colin Crossa4f08812018-10-02 22:03:40 -070019 "path/filepath"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080020 "reflect"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070021 "regexp"
Colin Crossb69301e2017-12-01 10:48:26 -080022 "sort"
Colin Crossd09b0b62018-04-18 11:06:47 -070023 "strings"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080024 "testing"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070025
26 "github.com/google/blueprint/proptools"
27
28 "android/soong/android"
29 "android/soong/cc"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080030)
31
32var (
33 resourceFiles = []string{
34 "res/layout/layout.xml",
35 "res/values/strings.xml",
36 "res/values-en-rUS/strings.xml",
37 }
38
39 compiledResourceFiles = []string{
40 "aapt2/res/layout_layout.xml.flat",
41 "aapt2/res/values_strings.arsc.flat",
42 "aapt2/res/values-en-rUS_strings.arsc.flat",
43 }
44)
45
Colin Cross98be1bb2019-12-13 20:41:13 -080046func testAppConfig(env map[string]string, bp string, fs map[string][]byte) android.Config {
Colin Cross527012a2017-11-30 22:56:16 -080047 appFS := map[string][]byte{}
48 for k, v := range fs {
49 appFS[k] = v
Colin Cross3bc7ffa2017-11-22 16:19:37 -080050 }
51
Colin Cross527012a2017-11-30 22:56:16 -080052 for _, file := range resourceFiles {
53 appFS[file] = nil
54 }
55
Colin Cross98be1bb2019-12-13 20:41:13 -080056 return testConfig(env, bp, appFS)
Colin Cross527012a2017-11-30 22:56:16 -080057}
58
59func testApp(t *testing.T, bp string) *android.TestContext {
Colin Cross98be1bb2019-12-13 20:41:13 -080060 config := testAppConfig(nil, bp, nil)
Colin Cross527012a2017-11-30 22:56:16 -080061
Colin Cross98be1bb2019-12-13 20:41:13 -080062 ctx := testContext()
Colin Cross527012a2017-11-30 22:56:16 -080063
64 run(t, ctx, config)
65
66 return ctx
Colin Cross3bc7ffa2017-11-22 16:19:37 -080067}
68
69func TestApp(t *testing.T) {
Colin Crossa97c5d32018-03-28 14:58:31 -070070 for _, moduleType := range []string{"android_app", "android_library"} {
71 t.Run(moduleType, func(t *testing.T) {
72 ctx := testApp(t, moduleType+` {
73 name: "foo",
74 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +090075 sdk_version: "current"
Colin Crossa97c5d32018-03-28 14:58:31 -070076 }
77 `)
Colin Cross3bc7ffa2017-11-22 16:19:37 -080078
Colin Crossa97c5d32018-03-28 14:58:31 -070079 foo := ctx.ModuleForTests("foo", "android_common")
Colin Cross3bc7ffa2017-11-22 16:19:37 -080080
Colin Cross31656952018-05-24 16:11:20 -070081 var expectedLinkImplicits []string
82
83 manifestFixer := foo.Output("manifest_fixer/AndroidManifest.xml")
84 expectedLinkImplicits = append(expectedLinkImplicits, manifestFixer.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080085
Colin Crossa97c5d32018-03-28 14:58:31 -070086 frameworkRes := ctx.ModuleForTests("framework-res", "android_common")
87 expectedLinkImplicits = append(expectedLinkImplicits,
88 frameworkRes.Output("package-res.apk").Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080089
Colin Crossa97c5d32018-03-28 14:58:31 -070090 // Test the mapping from input files to compiled output file names
91 compile := foo.Output(compiledResourceFiles[0])
92 if !reflect.DeepEqual(resourceFiles, compile.Inputs.Strings()) {
93 t.Errorf("expected aapt2 compile inputs expected:\n %#v\n got:\n %#v",
94 resourceFiles, compile.Inputs.Strings())
95 }
Colin Crossb69301e2017-12-01 10:48:26 -080096
Colin Crossa97c5d32018-03-28 14:58:31 -070097 compiledResourceOutputs := compile.Outputs.Strings()
98 sort.Strings(compiledResourceOutputs)
Colin Crossb69301e2017-12-01 10:48:26 -080099
Colin Crossa97c5d32018-03-28 14:58:31 -0700100 expectedLinkImplicits = append(expectedLinkImplicits, compiledResourceOutputs...)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800101
Colin Crossa97c5d32018-03-28 14:58:31 -0700102 list := foo.Output("aapt2/res.list")
103 expectedLinkImplicits = append(expectedLinkImplicits, list.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800104
Colin Crossa97c5d32018-03-28 14:58:31 -0700105 // Check that the link rule uses
106 res := ctx.ModuleForTests("foo", "android_common").Output("package-res.apk")
107 if !reflect.DeepEqual(expectedLinkImplicits, res.Implicits.Strings()) {
108 t.Errorf("expected aapt2 link implicits expected:\n %#v\n got:\n %#v",
109 expectedLinkImplicits, res.Implicits.Strings())
110 }
111 })
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800112 }
113}
Colin Cross890ff552017-11-30 20:13:19 -0800114
Colin Crosse560c4a2019-03-19 16:03:11 -0700115func TestAppSplits(t *testing.T) {
116 ctx := testApp(t, `
117 android_app {
118 name: "foo",
119 srcs: ["a.java"],
120 package_splits: ["v4", "v7,hdpi"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900121 sdk_version: "current"
Colin Crosse560c4a2019-03-19 16:03:11 -0700122 }`)
123
124 foo := ctx.ModuleForTests("foo", "android_common")
125
126 expectedOutputs := []string{
127 filepath.Join(buildDir, ".intermediates/foo/android_common/foo.apk"),
128 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v4.apk"),
129 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v7_hdpi.apk"),
130 }
131 for _, expectedOutput := range expectedOutputs {
132 foo.Output(expectedOutput)
133 }
134
Colin Cross41955e82019-05-29 14:40:35 -0700135 outputFiles, err := foo.Module().(*AndroidApp).OutputFiles("")
136 if err != nil {
137 t.Fatal(err)
138 }
139 if g, w := outputFiles.Strings(), expectedOutputs; !reflect.DeepEqual(g, w) {
140 t.Errorf(`want OutputFiles("") = %q, got %q`, w, g)
Colin Crosse560c4a2019-03-19 16:03:11 -0700141 }
142}
143
Sasha Smundaka7856c02020-04-23 09:49:59 -0700144func TestAndroidAppSet(t *testing.T) {
145 ctx, config := testJava(t, `
146 android_app_set {
147 name: "foo",
148 set: "prebuilts/apks/app.apks",
149 prerelease: true,
Jaewoong Jung11c1e0f2020-06-29 19:18:44 -0700150 }`)
Sasha Smundaka7856c02020-04-23 09:49:59 -0700151 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundak18d98bc2020-05-27 16:36:07 -0700152 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-04-23 09:49:59 -0700153 params := module.Output(packedSplitApks)
154 if params.Rule == nil {
155 t.Errorf("expected output %s is missing", packedSplitApks)
156 }
157 if s := params.Args["allow-prereleased"]; s != "true" {
158 t.Errorf("wrong allow-prereleased value: '%s', expected 'true'", s)
159 }
Jaewoong Jung11c1e0f2020-06-29 19:18:44 -0700160 if s := params.Args["partition"]; s != "system" {
161 t.Errorf("wrong partition value: '%s', expected 'system'", s)
162 }
Sasha Smundaka7856c02020-04-23 09:49:59 -0700163 mkEntries := android.AndroidMkEntriesForTest(t, config, "", module.Module())[0]
164 actualMaster := mkEntries.EntryMap["LOCAL_APK_SET_MASTER_FILE"]
165 expectedMaster := []string{"foo.apk"}
166 if !reflect.DeepEqual(actualMaster, expectedMaster) {
167 t.Errorf("Unexpected LOCAL_APK_SET_MASTER_FILE value: '%s', expected: '%s',",
168 actualMaster, expectedMaster)
169 }
170}
171
172func TestAndroidAppSet_Variants(t *testing.T) {
173 bp := `
174 android_app_set {
175 name: "foo",
176 set: "prebuilts/apks/app.apks",
177 }`
178 testCases := []struct {
179 name string
180 deviceArch *string
181 deviceSecondaryArch *string
182 aaptPrebuiltDPI []string
183 sdkVersion int
184 expected map[string]string
185 }{
186 {
187 name: "One",
188 deviceArch: proptools.StringPtr("x86"),
189 aaptPrebuiltDPI: []string{"ldpi", "xxhdpi"},
190 sdkVersion: 29,
191 expected: map[string]string{
192 "abis": "X86",
193 "allow-prereleased": "false",
194 "screen-densities": "LDPI,XXHDPI",
195 "sdk-version": "29",
196 "stem": "foo",
197 },
198 },
199 {
200 name: "Two",
201 deviceArch: proptools.StringPtr("x86_64"),
202 deviceSecondaryArch: proptools.StringPtr("x86"),
203 aaptPrebuiltDPI: nil,
204 sdkVersion: 30,
205 expected: map[string]string{
206 "abis": "X86_64,X86",
207 "allow-prereleased": "false",
208 "screen-densities": "all",
209 "sdk-version": "30",
210 "stem": "foo",
211 },
212 },
213 }
214
215 for _, test := range testCases {
216 config := testAppConfig(nil, bp, nil)
217 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
218 config.TestProductVariables.Platform_sdk_version = &test.sdkVersion
219 config.TestProductVariables.DeviceArch = test.deviceArch
220 config.TestProductVariables.DeviceSecondaryArch = test.deviceSecondaryArch
221 ctx := testContext()
222 run(t, ctx, config)
223 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundak18d98bc2020-05-27 16:36:07 -0700224 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-04-23 09:49:59 -0700225 params := module.Output(packedSplitApks)
226 for k, v := range test.expected {
227 if actual := params.Args[k]; actual != v {
228 t.Errorf("%s: bad build arg value for '%s': '%s', expected '%s'",
229 test.name, k, actual, v)
230 }
231 }
232 }
233}
234
Jeongik Cha538c0d02019-07-11 15:54:27 +0900235func TestPlatformAPIs(t *testing.T) {
236 testJava(t, `
237 android_app {
238 name: "foo",
239 srcs: ["a.java"],
240 platform_apis: true,
241 }
242 `)
243
244 testJava(t, `
245 android_app {
246 name: "foo",
247 srcs: ["a.java"],
248 sdk_version: "current",
249 }
250 `)
251
252 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
253 android_app {
254 name: "bar",
255 srcs: ["b.java"],
256 }
257 `)
258
259 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
260 android_app {
261 name: "bar",
262 srcs: ["b.java"],
263 sdk_version: "system_current",
264 platform_apis: true,
265 }
266 `)
267}
268
Jeongik Chae403e9e2019-12-07 00:16:24 +0900269func TestAndroidAppLinkType(t *testing.T) {
270 testJava(t, `
271 android_app {
272 name: "foo",
273 srcs: ["a.java"],
274 libs: ["bar"],
275 static_libs: ["baz"],
276 platform_apis: true,
277 }
278
279 java_library {
280 name: "bar",
281 sdk_version: "current",
282 srcs: ["b.java"],
283 }
284
285 android_library {
286 name: "baz",
287 sdk_version: "system_current",
288 srcs: ["c.java"],
289 }
290 `)
291
292 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.", `
293 android_app {
294 name: "foo",
295 srcs: ["a.java"],
296 libs: ["bar"],
297 sdk_version: "current",
298 static_libs: ["baz"],
299 }
300
301 java_library {
302 name: "bar",
303 sdk_version: "current",
304 srcs: ["b.java"],
305 }
306
307 android_library {
308 name: "baz",
309 sdk_version: "system_current",
310 srcs: ["c.java"],
311 }
312 `)
313
314 testJava(t, `
315 android_app {
316 name: "foo",
317 srcs: ["a.java"],
318 libs: ["bar"],
319 sdk_version: "system_current",
320 static_libs: ["baz"],
321 }
322
323 java_library {
324 name: "bar",
325 sdk_version: "current",
326 srcs: ["b.java"],
327 }
328
329 android_library {
330 name: "baz",
331 sdk_version: "system_current",
332 srcs: ["c.java"],
333 }
334 `)
335
336 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.", `
337 android_app {
338 name: "foo",
339 srcs: ["a.java"],
340 libs: ["bar"],
341 sdk_version: "system_current",
342 static_libs: ["baz"],
343 }
344
345 java_library {
346 name: "bar",
347 sdk_version: "current",
348 srcs: ["b.java"],
349 }
350
351 android_library {
352 name: "baz",
353 srcs: ["c.java"],
354 }
355 `)
356}
357
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100358func TestUpdatableApps(t *testing.T) {
359 testCases := []struct {
360 name string
361 bp string
362 expectedError string
363 }{
364 {
365 name: "Stable public SDK",
366 bp: `android_app {
367 name: "foo",
368 srcs: ["a.java"],
369 sdk_version: "29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100370 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100371 updatable: true,
372 }`,
373 },
374 {
375 name: "Stable system SDK",
376 bp: `android_app {
377 name: "foo",
378 srcs: ["a.java"],
379 sdk_version: "system_29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100380 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100381 updatable: true,
382 }`,
383 },
384 {
385 name: "Current public SDK",
386 bp: `android_app {
387 name: "foo",
388 srcs: ["a.java"],
389 sdk_version: "current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100390 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100391 updatable: true,
392 }`,
393 },
394 {
395 name: "Current system SDK",
396 bp: `android_app {
397 name: "foo",
398 srcs: ["a.java"],
399 sdk_version: "system_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100400 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100401 updatable: true,
402 }`,
403 },
404 {
405 name: "Current module SDK",
406 bp: `android_app {
407 name: "foo",
408 srcs: ["a.java"],
409 sdk_version: "module_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100410 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100411 updatable: true,
412 }`,
413 },
414 {
415 name: "Current core SDK",
416 bp: `android_app {
417 name: "foo",
418 srcs: ["a.java"],
419 sdk_version: "core_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100420 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100421 updatable: true,
422 }`,
423 },
424 {
425 name: "No Platform APIs",
426 bp: `android_app {
427 name: "foo",
428 srcs: ["a.java"],
429 platform_apis: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100430 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100431 updatable: true,
432 }`,
433 expectedError: "Updatable apps must use stable SDKs",
434 },
435 {
436 name: "No Core Platform APIs",
437 bp: `android_app {
438 name: "foo",
439 srcs: ["a.java"],
440 sdk_version: "core_platform",
Artur Satayevf40fc852020-04-16 13:43:02 +0100441 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100442 updatable: true,
443 }`,
444 expectedError: "Updatable apps must use stable SDKs",
445 },
446 {
447 name: "No unspecified APIs",
448 bp: `android_app {
449 name: "foo",
450 srcs: ["a.java"],
451 updatable: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100452 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100453 }`,
454 expectedError: "Updatable apps must use stable SDK",
455 },
Artur Satayevf40fc852020-04-16 13:43:02 +0100456 {
457 name: "Must specify min_sdk_version",
458 bp: `android_app {
459 name: "app_without_min_sdk_version",
460 srcs: ["a.java"],
461 sdk_version: "29",
462 updatable: true,
463 }`,
464 expectedError: "updatable apps must set min_sdk_version.",
465 },
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100466 }
467
468 for _, test := range testCases {
469 t.Run(test.name, func(t *testing.T) {
470 if test.expectedError == "" {
471 testJava(t, test.bp)
472 } else {
473 testJavaError(t, test.expectedError, test.bp)
474 }
475 })
476 }
477}
478
Jooyung Han749dc692020-04-15 11:03:39 +0900479func TestUpdatableApps_TransitiveDepsShouldSetMinSdkVersion(t *testing.T) {
480 testJavaError(t, `module "bar".*: should support min_sdk_version\(29\)`, cc.GatherRequiredDepsForTest(android.Android)+`
481 android_app {
482 name: "foo",
483 srcs: ["a.java"],
484 updatable: true,
485 sdk_version: "current",
486 min_sdk_version: "29",
487 static_libs: ["bar"],
488 }
489
490 java_library {
491 name: "bar",
492 sdk_version: "current",
493 }
494 `)
495}
496
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900497func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
498 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
499 android_app {
500 name: "foo",
501 srcs: ["a.java"],
502 updatable: true,
503 sdk_version: "current",
504 min_sdk_version: "current",
505 jni_libs: ["libjni"],
506 }
507
508 cc_library {
509 name: "libjni",
510 stl: "none",
511 system_shared_libs: [],
512 sdk_version: "current",
513 }
514 `)
515}
516
517func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
518 bp := cc.GatherRequiredDepsForTest(android.Android) + `
519 android_app {
520 name: "foo",
521 srcs: ["a.java"],
522 updatable: true,
523 sdk_version: "current",
524 min_sdk_version: "29",
525 jni_libs: ["libjni"],
526 }
527
528 cc_library {
529 name: "libjni",
530 stl: "none",
531 system_shared_libs: [],
532 sdk_version: "29",
533 }
534
535 ndk_prebuilt_object {
536 name: "ndk_crtbegin_so.29",
537 sdk_version: "29",
538 }
539
540 ndk_prebuilt_object {
541 name: "ndk_crtend_so.29",
542 sdk_version: "29",
543 }
544 `
545 fs := map[string][]byte{
546 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
547 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
548 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
549 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
550 }
551
552 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
553
554 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
555 var crtbeginFound, crtendFound bool
556 for _, input := range inputs {
557 switch input.String() {
558 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o":
559 crtbeginFound = true
560 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o":
561 crtendFound = true
562 }
563 }
564 if !crtbeginFound || !crtendFound {
565 t.Error("should link with ndk_crtbegin_so.29 and ndk_crtend_so.29")
566 }
567}
568
569func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
570 bp := cc.GatherRequiredDepsForTest(android.Android) + `
571 android_app {
572 name: "foo",
573 srcs: ["a.java"],
574 updatable: true,
575 sdk_version: "current",
576 min_sdk_version: "29", // this APK should support 29
577 jni_libs: ["libjni"],
578 }
579
580 cc_library {
581 name: "libjni",
582 stl: "none",
583 sdk_version: "current",
584 }
585 `
586 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
587}
588
589func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
590 bp := cc.GatherRequiredDepsForTest(android.Android) + `
591 android_app {
592 name: "foo",
593 srcs: ["a.java"],
594 updatable: true,
595 sdk_version: "current",
596 min_sdk_version: "29", // this APK should support 29
597 jni_libs: ["libjni"],
598 }
599
600 cc_library {
601 name: "libjni",
602 stl: "none",
603 shared_libs: ["libbar"],
604 system_shared_libs: [],
605 sdk_version: "27",
606 }
607
608 cc_library {
609 name: "libbar",
610 stl: "none",
611 system_shared_libs: [],
612 sdk_version: "current",
613 }
614 `
615 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
616}
617
Colin Cross0ddae7f2019-02-07 15:30:01 -0800618func TestResourceDirs(t *testing.T) {
619 testCases := []struct {
620 name string
621 prop string
622 resources []string
623 }{
624 {
625 name: "no resource_dirs",
626 prop: "",
627 resources: []string{"res/res/values/strings.xml"},
628 },
629 {
630 name: "resource_dirs",
631 prop: `resource_dirs: ["res"]`,
632 resources: []string{"res/res/values/strings.xml"},
633 },
634 {
635 name: "empty resource_dirs",
636 prop: `resource_dirs: []`,
637 resources: nil,
638 },
639 }
640
641 fs := map[string][]byte{
642 "res/res/values/strings.xml": nil,
643 }
644
645 bp := `
646 android_app {
647 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900648 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800649 %s
650 }
651 `
652
653 for _, testCase := range testCases {
654 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800655 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
656 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800657 run(t, ctx, config)
658
659 module := ctx.ModuleForTests("foo", "android_common")
660 resourceList := module.MaybeOutput("aapt2/res.list")
661
662 var resources []string
663 if resourceList.Rule != nil {
664 for _, compiledResource := range resourceList.Inputs.Strings() {
665 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
666 }
667 }
668
669 if !reflect.DeepEqual(resources, testCase.resources) {
670 t.Errorf("expected resource files %q, got %q",
671 testCase.resources, resources)
672 }
673 })
674 }
675}
676
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800677func TestLibraryAssets(t *testing.T) {
678 bp := `
679 android_app {
680 name: "foo",
681 sdk_version: "current",
682 static_libs: ["lib1", "lib2", "lib3"],
683 }
684
685 android_library {
686 name: "lib1",
687 sdk_version: "current",
688 asset_dirs: ["assets_a"],
689 }
690
691 android_library {
692 name: "lib2",
693 sdk_version: "current",
694 }
695
696 android_library {
697 name: "lib3",
698 sdk_version: "current",
699 static_libs: ["lib4"],
700 }
701
702 android_library {
703 name: "lib4",
704 sdk_version: "current",
705 asset_dirs: ["assets_b"],
706 }
707 `
708
709 testCases := []struct {
710 name string
711 assetFlag string
712 assetPackages []string
713 }{
714 {
715 name: "foo",
716 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
717 assetPackages: []string{
718 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
719 buildDir + "/.intermediates/lib1/android_common/assets.zip",
720 buildDir + "/.intermediates/lib3/android_common/assets.zip",
721 },
722 },
723 {
724 name: "lib1",
725 assetFlag: "-A assets_a",
726 },
727 {
728 name: "lib2",
729 },
730 {
731 name: "lib3",
732 assetPackages: []string{
733 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
734 buildDir + "/.intermediates/lib4/android_common/assets.zip",
735 },
736 },
737 {
738 name: "lib4",
739 assetFlag: "-A assets_b",
740 },
741 }
742 ctx := testApp(t, bp)
743
744 for _, test := range testCases {
745 t.Run(test.name, func(t *testing.T) {
746 m := ctx.ModuleForTests(test.name, "android_common")
747
748 // Check asset flag in aapt2 link flags
749 var aapt2link android.TestingBuildParams
750 if len(test.assetPackages) > 0 {
751 aapt2link = m.Output("aapt2/package-res.apk")
752 } else {
753 aapt2link = m.Output("package-res.apk")
754 }
755 aapt2Flags := aapt2link.Args["flags"]
756 if test.assetFlag != "" {
757 if !strings.Contains(aapt2Flags, test.assetFlag) {
758 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
759 }
760 } else {
761 if strings.Contains(aapt2Flags, " -A ") {
762 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
763 }
764 }
765
766 // Check asset merge rule.
767 if len(test.assetPackages) > 0 {
768 mergeAssets := m.Output("package-res.apk")
769 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
770 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
771 mergeAssets.Inputs.Strings(), test.assetPackages)
772 }
773 }
774 })
775 }
776}
777
Colin Crossbec85302019-02-13 13:15:46 -0800778func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800779 testCases := []struct {
780 name string
781 enforceRROTargets []string
782 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800783 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800784 overlayFiles map[string][]string
785 rroDirs map[string][]string
786 }{
787 {
788 name: "no RRO",
789 enforceRROTargets: nil,
790 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800791 resourceFiles: map[string][]string{
792 "foo": nil,
793 "bar": {"bar/res/res/values/strings.xml"},
794 "lib": nil,
795 "lib2": {"lib2/res/res/values/strings.xml"},
796 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800797 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800798 "foo": {
799 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800800 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000801 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800802 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800803 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
804 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000805 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800806 },
Colin Crossbec85302019-02-13 13:15:46 -0800807 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800808 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
809 "device/vendor/blah/overlay/bar/res/values/strings.xml",
810 },
Colin Crossbec85302019-02-13 13:15:46 -0800811 "lib": {
812 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
813 "lib/res/res/values/strings.xml",
814 "device/vendor/blah/overlay/lib/res/values/strings.xml",
815 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800816 },
817 rroDirs: map[string][]string{
818 "foo": nil,
819 "bar": nil,
820 },
821 },
822 {
823 name: "enforce RRO on foo",
824 enforceRROTargets: []string{"foo"},
825 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800826 resourceFiles: map[string][]string{
827 "foo": nil,
828 "bar": {"bar/res/res/values/strings.xml"},
829 "lib": nil,
830 "lib2": {"lib2/res/res/values/strings.xml"},
831 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800832 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800833 "foo": {
834 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800835 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000836 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800837 "foo/res/res/values/strings.xml",
838 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
839 },
Colin Crossbec85302019-02-13 13:15:46 -0800840 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800841 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
842 "device/vendor/blah/overlay/bar/res/values/strings.xml",
843 },
Colin Crossbec85302019-02-13 13:15:46 -0800844 "lib": {
845 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
846 "lib/res/res/values/strings.xml",
847 "device/vendor/blah/overlay/lib/res/values/strings.xml",
848 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800849 },
Colin Crossc1c37552019-01-31 11:42:41 -0800850
Colin Cross5c4791c2019-02-01 11:44:44 -0800851 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800852 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000853 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800854 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
855 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000856 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800857 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800858 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800859 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800860 },
861 },
862 {
863 name: "enforce RRO on all",
864 enforceRROTargets: []string{"*"},
865 enforceRROExcludedOverlays: []string{
866 // Excluding specific apps/res directories also allowed.
867 "device/vendor/blah/static_overlay/foo",
868 "device/vendor/blah/static_overlay/bar/res",
869 },
Colin Crossbec85302019-02-13 13:15:46 -0800870 resourceFiles: map[string][]string{
871 "foo": nil,
872 "bar": {"bar/res/res/values/strings.xml"},
873 "lib": nil,
874 "lib2": {"lib2/res/res/values/strings.xml"},
875 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800876 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800877 "foo": {
878 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800879 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000880 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800881 "foo/res/res/values/strings.xml",
882 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
883 },
Colin Crossbec85302019-02-13 13:15:46 -0800884 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
885 "lib": {
886 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
887 "lib/res/res/values/strings.xml",
888 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800889 },
890 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800891 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000892 "device:device/vendor/blah/overlay/foo/res",
893 "product:product/vendor/blah/overlay/foo/res",
894 // Lib dep comes after the direct deps
895 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800896 },
Anton Hansson53c88442019-03-18 15:53:16 +0000897 "bar": {"device:device/vendor/blah/overlay/bar/res"},
898 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800899 },
900 },
901 }
902
Anton Hansson53c88442019-03-18 15:53:16 +0000903 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800904 "device/vendor/blah/overlay",
905 "device/vendor/blah/overlay2",
906 "device/vendor/blah/static_overlay",
907 }
908
Anton Hansson53c88442019-03-18 15:53:16 +0000909 productResourceOverlays := []string{
910 "product/vendor/blah/overlay",
911 }
912
Colin Cross890ff552017-11-30 20:13:19 -0800913 fs := map[string][]byte{
914 "foo/res/res/values/strings.xml": nil,
915 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800916 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800917 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800918 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
919 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800920 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800921 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
922 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
923 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000924 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800925 }
926
927 bp := `
928 android_app {
929 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900930 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800931 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000932 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800933 }
934
935 android_app {
936 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900937 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800938 resource_dirs: ["bar/res"],
939 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800940
941 android_library {
942 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900943 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800944 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800945 static_libs: ["lib2"],
946 }
947
948 android_library {
949 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900950 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800951 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800952 }
Anton Hansson53c88442019-03-18 15:53:16 +0000953
954 // This library has the same resources as lib (should not lead to dupe RROs)
955 android_library {
956 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900957 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000958 resource_dirs: ["lib/res"]
959 }
Colin Cross890ff552017-11-30 20:13:19 -0800960 `
961
Colin Cross5c4791c2019-02-01 11:44:44 -0800962 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800963 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800964 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000965 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
966 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800967 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800968 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800969 }
970 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800971 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800972 }
973
Colin Cross98be1bb2019-12-13 20:41:13 -0800974 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800975 run(t, ctx, config)
976
Colin Crossbec85302019-02-13 13:15:46 -0800977 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
978 for _, o := range list {
979 res := module.MaybeOutput(o)
980 if res.Rule != nil {
981 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
982 // verify the inputs to the .arsc.flat rule.
983 files = append(files, res.Inputs.Strings()...)
984 } else {
985 // Otherwise, verify the full path to the output of the other module
986 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000987 }
Colin Cross890ff552017-11-30 20:13:19 -0800988 }
Colin Crossbec85302019-02-13 13:15:46 -0800989 return files
Colin Cross890ff552017-11-30 20:13:19 -0800990 }
991
Colin Crossbec85302019-02-13 13:15:46 -0800992 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
993 module := ctx.ModuleForTests(moduleName, "android_common")
994 resourceList := module.MaybeOutput("aapt2/res.list")
995 if resourceList.Rule != nil {
996 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000997 }
Colin Crossbec85302019-02-13 13:15:46 -0800998 overlayList := module.MaybeOutput("aapt2/overlay.list")
999 if overlayList.Rule != nil {
1000 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
1001 }
1002
Anton Hansson53c88442019-03-18 15:53:16 +00001003 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
1004 var prefix string
1005 if d.overlayType == device {
1006 prefix = "device:"
1007 } else if d.overlayType == product {
1008 prefix = "product:"
1009 } else {
1010 t.Fatalf("Unexpected overlayType %d", d.overlayType)
1011 }
1012 rroDirs = append(rroDirs, prefix+d.path.String())
1013 }
Colin Crossbec85302019-02-13 13:15:46 -08001014
1015 return resourceFiles, overlayFiles, rroDirs
1016 }
1017
1018 modules := []string{"foo", "bar", "lib", "lib2"}
1019 for _, module := range modules {
1020 resourceFiles, overlayFiles, rroDirs := getResources(module)
1021
1022 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
1023 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
1024 module, testCase.resourceFiles[module], resourceFiles)
1025 }
1026 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
1027 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
1028 module, testCase.overlayFiles[module], overlayFiles)
1029 }
1030 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +00001031 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -08001032 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +00001033 }
Colin Cross890ff552017-11-30 20:13:19 -08001034 }
Colin Cross890ff552017-11-30 20:13:19 -08001035 })
1036 }
1037}
Colin Crossd09b0b62018-04-18 11:06:47 -07001038
1039func TestAppSdkVersion(t *testing.T) {
1040 testCases := []struct {
1041 name string
1042 sdkVersion string
1043 platformSdkInt int
1044 platformSdkCodename string
1045 platformSdkFinal bool
1046 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +09001047 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -07001048 }{
1049 {
1050 name: "current final SDK",
1051 sdkVersion: "current",
1052 platformSdkInt: 27,
1053 platformSdkCodename: "REL",
1054 platformSdkFinal: true,
1055 expectedMinSdkVersion: "27",
1056 },
1057 {
1058 name: "current non-final SDK",
1059 sdkVersion: "current",
1060 platformSdkInt: 27,
1061 platformSdkCodename: "OMR1",
1062 platformSdkFinal: false,
1063 expectedMinSdkVersion: "OMR1",
1064 },
1065 {
1066 name: "default final SDK",
1067 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001068 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001069 platformSdkInt: 27,
1070 platformSdkCodename: "REL",
1071 platformSdkFinal: true,
1072 expectedMinSdkVersion: "27",
1073 },
1074 {
1075 name: "default non-final SDK",
1076 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001077 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001078 platformSdkInt: 27,
1079 platformSdkCodename: "OMR1",
1080 platformSdkFinal: false,
1081 expectedMinSdkVersion: "OMR1",
1082 },
1083 {
1084 name: "14",
1085 sdkVersion: "14",
1086 expectedMinSdkVersion: "14",
1087 },
1088 }
1089
1090 for _, moduleType := range []string{"android_app", "android_library"} {
1091 for _, test := range testCases {
1092 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001093 platformApiProp := ""
1094 if test.platformApis {
1095 platformApiProp = "platform_apis: true,"
1096 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001097 bp := fmt.Sprintf(`%s {
1098 name: "foo",
1099 srcs: ["a.java"],
1100 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001101 %s
1102 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001103
Colin Cross98be1bb2019-12-13 20:41:13 -08001104 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001105 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1106 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1107 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1108
Colin Cross98be1bb2019-12-13 20:41:13 -08001109 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -07001110
1111 run(t, ctx, config)
1112
1113 foo := ctx.ModuleForTests("foo", "android_common")
1114 link := foo.Output("package-res.apk")
1115 linkFlags := strings.Split(link.Args["flags"], " ")
1116 min := android.IndexList("--min-sdk-version", linkFlags)
1117 target := android.IndexList("--target-sdk-version", linkFlags)
1118
1119 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1120 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1121 }
1122
1123 gotMinSdkVersion := linkFlags[min+1]
1124 gotTargetSdkVersion := linkFlags[target+1]
1125
1126 if gotMinSdkVersion != test.expectedMinSdkVersion {
1127 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1128 test.expectedMinSdkVersion, gotMinSdkVersion)
1129 }
1130
1131 if gotTargetSdkVersion != test.expectedMinSdkVersion {
1132 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1133 test.expectedMinSdkVersion, gotTargetSdkVersion)
1134 }
1135 })
1136 }
1137 }
1138}
Colin Crossa4f08812018-10-02 22:03:40 -07001139
Paul Duffin50c217c2019-06-12 13:25:22 +01001140func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001141 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001142 cc_library {
1143 name: "libjni",
1144 system_shared_libs: [],
Colin Crossc511bc52020-04-07 16:50:32 +00001145 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001146 stl: "none",
1147 }
1148
1149 android_test {
1150 name: "test",
1151 sdk_version: "core_platform",
1152 jni_libs: ["libjni"],
1153 }
1154
1155 android_test {
1156 name: "test_first",
1157 sdk_version: "core_platform",
1158 compile_multilib: "first",
1159 jni_libs: ["libjni"],
1160 }
1161
1162 android_test {
1163 name: "test_both",
1164 sdk_version: "core_platform",
1165 compile_multilib: "both",
1166 jni_libs: ["libjni"],
1167 }
1168
1169 android_test {
1170 name: "test_32",
1171 sdk_version: "core_platform",
1172 compile_multilib: "32",
1173 jni_libs: ["libjni"],
1174 }
1175
1176 android_test {
1177 name: "test_64",
1178 sdk_version: "core_platform",
1179 compile_multilib: "64",
1180 jni_libs: ["libjni"],
1181 }
1182 `)
1183
1184 testCases := []struct {
1185 name string
1186 abis []string
1187 }{
1188 {"test", []string{"arm64-v8a"}},
1189 {"test_first", []string{"arm64-v8a"}},
1190 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1191 {"test_32", []string{"armeabi-v7a"}},
1192 {"test_64", []string{"arm64-v8a"}},
1193 }
1194
1195 for _, test := range testCases {
1196 t.Run(test.name, func(t *testing.T) {
1197 app := ctx.ModuleForTests(test.name, "android_common")
1198 jniLibZip := app.Output("jnilibs.zip")
1199 var abis []string
1200 args := strings.Fields(jniLibZip.Args["jarArgs"])
1201 for i := 0; i < len(args); i++ {
1202 if args[i] == "-P" {
1203 abis = append(abis, filepath.Base(args[i+1]))
1204 i++
1205 }
1206 }
1207 if !reflect.DeepEqual(abis, test.abis) {
1208 t.Errorf("want abis %v, got %v", test.abis, abis)
1209 }
1210 })
1211 }
1212}
1213
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001214func TestAppSdkVersionByPartition(t *testing.T) {
1215 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1216 android_app {
1217 name: "foo",
1218 srcs: ["a.java"],
1219 vendor: true,
1220 platform_apis: true,
1221 }
1222 `)
1223
1224 testJava(t, `
1225 android_app {
1226 name: "bar",
1227 srcs: ["b.java"],
1228 platform_apis: true,
1229 }
1230 `)
1231
1232 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001233 bp := `
1234 android_app {
1235 name: "foo",
1236 srcs: ["a.java"],
1237 product_specific: true,
1238 platform_apis: true,
1239 }
1240 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001241
1242 config := testAppConfig(nil, bp, nil)
1243 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001244 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001245 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 +09001246 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001247 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001248 }
1249 }
1250}
1251
Paul Duffin50c217c2019-06-12 13:25:22 +01001252func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001253 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001254 cc_library {
1255 name: "libjni",
1256 system_shared_libs: [],
1257 stl: "none",
Colin Cross094cde42020-02-15 10:38:00 -08001258 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001259 }
1260
1261 android_app {
1262 name: "app",
1263 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001264 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001265 }
1266
1267 android_app {
1268 name: "app_noembed",
1269 jni_libs: ["libjni"],
1270 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001271 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001272 }
1273
1274 android_app {
1275 name: "app_embed",
1276 jni_libs: ["libjni"],
1277 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001278 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001279 }
1280
1281 android_test {
1282 name: "test",
Colin Crossc511bc52020-04-07 16:50:32 +00001283 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001284 jni_libs: ["libjni"],
1285 }
1286
1287 android_test {
1288 name: "test_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001289 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001290 jni_libs: ["libjni"],
1291 use_embedded_native_libs: false,
1292 }
1293
1294 android_test_helper_app {
1295 name: "test_helper",
Colin Crossc511bc52020-04-07 16:50:32 +00001296 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001297 jni_libs: ["libjni"],
1298 }
1299
1300 android_test_helper_app {
1301 name: "test_helper_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001302 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001303 jni_libs: ["libjni"],
1304 use_embedded_native_libs: false,
1305 }
1306 `)
1307
1308 testCases := []struct {
1309 name string
1310 packaged bool
1311 compressed bool
1312 }{
1313 {"app", false, false},
1314 {"app_noembed", false, false},
1315 {"app_embed", true, false},
1316 {"test", true, false},
1317 {"test_noembed", true, true},
1318 {"test_helper", true, false},
1319 {"test_helper_noembed", true, true},
1320 }
1321
1322 for _, test := range testCases {
1323 t.Run(test.name, func(t *testing.T) {
1324 app := ctx.ModuleForTests(test.name, "android_common")
1325 jniLibZip := app.MaybeOutput("jnilibs.zip")
1326 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1327 t.Errorf("expected jni packaged %v, got %v", w, g)
1328 }
1329
1330 if jniLibZip.Rule != nil {
1331 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1332 t.Errorf("expected jni compressed %v, got %v", w, g)
1333 }
Colin Crossc511bc52020-04-07 16:50:32 +00001334
1335 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1336 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1337 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001338 }
1339 })
1340 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001341}
1342
Colin Cross3c007702020-05-08 11:20:24 -07001343func TestJNISDK(t *testing.T) {
1344 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1345 cc_library {
1346 name: "libjni",
1347 system_shared_libs: [],
1348 stl: "none",
1349 sdk_version: "current",
1350 }
1351
1352 android_test {
1353 name: "app_platform",
1354 jni_libs: ["libjni"],
1355 platform_apis: true,
1356 }
1357
1358 android_test {
1359 name: "app_sdk",
1360 jni_libs: ["libjni"],
1361 sdk_version: "current",
1362 }
1363
1364 android_test {
1365 name: "app_force_platform",
1366 jni_libs: ["libjni"],
1367 sdk_version: "current",
1368 jni_uses_platform_apis: true,
1369 }
1370
1371 android_test {
1372 name: "app_force_sdk",
1373 jni_libs: ["libjni"],
1374 platform_apis: true,
1375 jni_uses_sdk_apis: true,
1376 }
Colin Crossc2d24052020-05-13 11:05:02 -07001377
1378 cc_library {
1379 name: "libvendorjni",
1380 system_shared_libs: [],
1381 stl: "none",
1382 vendor: true,
1383 }
1384
1385 android_test {
1386 name: "app_vendor",
1387 jni_libs: ["libvendorjni"],
1388 sdk_version: "current",
1389 vendor: true,
1390 }
Colin Cross3c007702020-05-08 11:20:24 -07001391 `)
1392
1393 testCases := []struct {
Colin Crossc2d24052020-05-13 11:05:02 -07001394 name string
1395 sdkJNI bool
1396 vendorJNI bool
Colin Cross3c007702020-05-08 11:20:24 -07001397 }{
Colin Crossc2d24052020-05-13 11:05:02 -07001398 {name: "app_platform"},
1399 {name: "app_sdk", sdkJNI: true},
1400 {name: "app_force_platform"},
1401 {name: "app_force_sdk", sdkJNI: true},
1402 {name: "app_vendor", vendorJNI: true},
Colin Cross3c007702020-05-08 11:20:24 -07001403 }
1404
Colin Crossc2d24052020-05-13 11:05:02 -07001405 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1406 Output("libjni.so").Output.String()
1407 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1408 Output("libjni.so").Output.String()
1409 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1410 Output("libvendorjni.so").Output.String()
1411
Colin Cross3c007702020-05-08 11:20:24 -07001412 for _, test := range testCases {
1413 t.Run(test.name, func(t *testing.T) {
1414 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross3c007702020-05-08 11:20:24 -07001415
1416 jniLibZip := app.MaybeOutput("jnilibs.zip")
1417 if len(jniLibZip.Implicits) != 1 {
1418 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1419 }
1420 gotJNI := jniLibZip.Implicits[0].String()
1421
1422 if test.sdkJNI {
1423 if gotJNI != sdkJNI {
1424 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1425 }
Colin Crossc2d24052020-05-13 11:05:02 -07001426 } else if test.vendorJNI {
1427 if gotJNI != vendorJNI {
1428 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1429 }
Colin Cross3c007702020-05-08 11:20:24 -07001430 } else {
1431 if gotJNI != platformJNI {
1432 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1433 }
1434 }
1435 })
1436 }
1437
1438 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1439 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1440 android_test {
1441 name: "app_platform",
1442 platform_apis: true,
1443 jni_uses_platform_apis: true,
1444 }
1445 `)
1446 })
1447
1448 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1449 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1450 android_test {
1451 name: "app_sdk",
1452 sdk_version: "current",
1453 jni_uses_sdk_apis: true,
1454 }
1455 `)
1456 })
1457
1458}
1459
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001460func TestCertificates(t *testing.T) {
1461 testCases := []struct {
1462 name string
1463 bp string
1464 certificateOverride string
Liz Kammere2b27f42020-05-07 13:24:05 -07001465 expectedLineage string
1466 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001467 }{
1468 {
1469 name: "default",
1470 bp: `
1471 android_app {
1472 name: "foo",
1473 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001474 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001475 }
1476 `,
1477 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001478 expectedLineage: "",
1479 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001480 },
1481 {
1482 name: "module certificate property",
1483 bp: `
1484 android_app {
1485 name: "foo",
1486 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001487 certificate: ":new_certificate",
1488 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001489 }
1490
1491 android_app_certificate {
1492 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001493 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001494 }
1495 `,
1496 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001497 expectedLineage: "",
1498 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001499 },
1500 {
1501 name: "path certificate property",
1502 bp: `
1503 android_app {
1504 name: "foo",
1505 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001506 certificate: "expiredkey",
1507 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001508 }
1509 `,
1510 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001511 expectedLineage: "",
1512 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001513 },
1514 {
1515 name: "certificate overrides",
1516 bp: `
1517 android_app {
1518 name: "foo",
1519 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001520 certificate: "expiredkey",
1521 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001522 }
1523
1524 android_app_certificate {
1525 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001526 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001527 }
1528 `,
1529 certificateOverride: "foo:new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001530 expectedLineage: "",
1531 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1532 },
1533 {
1534 name: "certificate lineage",
1535 bp: `
1536 android_app {
1537 name: "foo",
1538 srcs: ["a.java"],
1539 certificate: ":new_certificate",
1540 lineage: "lineage.bin",
1541 sdk_version: "current",
1542 }
1543
1544 android_app_certificate {
1545 name: "new_certificate",
1546 certificate: "cert/new_cert",
1547 }
1548 `,
1549 certificateOverride: "",
1550 expectedLineage: "--lineage lineage.bin",
1551 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001552 },
1553 }
1554
1555 for _, test := range testCases {
1556 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001557 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001558 if test.certificateOverride != "" {
1559 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1560 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001561 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001562
1563 run(t, ctx, config)
1564 foo := ctx.ModuleForTests("foo", "android_common")
1565
1566 signapk := foo.Output("foo.apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001567 signCertificateFlags := signapk.Args["certificates"]
1568 if test.expectedCertificate != signCertificateFlags {
1569 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1570 }
1571
1572 signFlags := signapk.Args["flags"]
1573 if test.expectedLineage != signFlags {
1574 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001575 }
1576 })
1577 }
1578}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001579
1580func TestPackageNameOverride(t *testing.T) {
1581 testCases := []struct {
1582 name string
1583 bp string
1584 packageNameOverride string
1585 expected []string
1586 }{
1587 {
1588 name: "default",
1589 bp: `
1590 android_app {
1591 name: "foo",
1592 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001593 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001594 }
1595 `,
1596 packageNameOverride: "",
1597 expected: []string{
1598 buildDir + "/.intermediates/foo/android_common/foo.apk",
1599 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1600 },
1601 },
1602 {
1603 name: "overridden",
1604 bp: `
1605 android_app {
1606 name: "foo",
1607 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001608 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001609 }
1610 `,
1611 packageNameOverride: "foo:bar",
1612 expected: []string{
1613 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001614 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001615 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1616 },
1617 },
1618 }
1619
1620 for _, test := range testCases {
1621 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001622 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001623 if test.packageNameOverride != "" {
1624 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1625 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001626 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001627
1628 run(t, ctx, config)
1629 foo := ctx.ModuleForTests("foo", "android_common")
1630
1631 outputs := foo.AllOutputs()
1632 outputMap := make(map[string]bool)
1633 for _, o := range outputs {
1634 outputMap[o] = true
1635 }
1636 for _, e := range test.expected {
1637 if _, exist := outputMap[e]; !exist {
1638 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1639 }
1640 }
1641 })
1642 }
1643}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001644
1645func TestInstrumentationTargetOverridden(t *testing.T) {
1646 bp := `
1647 android_app {
1648 name: "foo",
1649 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001650 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001651 }
1652
1653 android_test {
1654 name: "bar",
1655 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001656 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001657 }
1658 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001659 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001660 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001661 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001662
1663 run(t, ctx, config)
1664
1665 bar := ctx.ModuleForTests("bar", "android_common")
1666 res := bar.Output("package-res.apk")
1667 aapt2Flags := res.Args["flags"]
1668 e := "--rename-instrumentation-target-package org.dandroid.bp"
1669 if !strings.Contains(aapt2Flags, e) {
1670 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1671 }
1672}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001673
1674func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001675 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001676 android_app {
1677 name: "foo",
1678 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001679 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001680 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001681 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001682 }
1683
1684 override_android_app {
1685 name: "bar",
1686 base: "foo",
1687 certificate: ":new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001688 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001689 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001690 }
1691
1692 android_app_certificate {
1693 name: "new_certificate",
1694 certificate: "cert/new_cert",
1695 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001696
1697 override_android_app {
1698 name: "baz",
1699 base: "foo",
1700 package_name: "org.dandroid.bp",
1701 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001702 `)
1703
1704 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001705 moduleName string
1706 variantName string
1707 apkName string
1708 apkPath string
Liz Kammere2b27f42020-05-07 13:24:05 -07001709 certFlag string
1710 lineageFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001711 overrides []string
Liz Kammer1d5983b2020-05-19 19:15:37 +00001712 aaptFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001713 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001714 }{
1715 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001716 moduleName: "foo",
1717 variantName: "android_common",
1718 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001719 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1720 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001721 overrides: []string{"qux"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001722 aaptFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001723 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001724 },
1725 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001726 moduleName: "bar",
1727 variantName: "android_common_bar",
1728 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001729 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1730 lineageFlag: "--lineage lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001731 overrides: []string{"qux", "foo"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001732 aaptFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001733 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001734 },
1735 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001736 moduleName: "baz",
1737 variantName: "android_common_baz",
1738 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001739 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1740 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001741 overrides: []string{"qux", "foo"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001742 aaptFlag: "--rename-manifest-package org.dandroid.bp",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001743 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001744 },
1745 }
1746 for _, expected := range expectedVariants {
1747 variant := ctx.ModuleForTests("foo", expected.variantName)
1748
1749 // Check the final apk name
1750 outputs := variant.AllOutputs()
1751 expectedApkPath := buildDir + expected.apkPath
1752 found := false
1753 for _, o := range outputs {
1754 if o == expectedApkPath {
1755 found = true
1756 break
1757 }
1758 }
1759 if !found {
1760 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1761 }
1762
1763 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001764 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001765 certFlag := signapk.Args["certificates"]
1766 if expected.certFlag != certFlag {
1767 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1768 }
1769
1770 // Check the lineage flags
1771 lineageFlag := signapk.Args["flags"]
1772 if expected.lineageFlag != lineageFlag {
1773 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001774 }
1775
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001776 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001777 mod := variant.Module().(*AndroidApp)
1778 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1779 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1780 expected.overrides, mod.appProperties.Overrides)
1781 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001782
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001783 // Test Overridable property: Logging_parent
1784 logging_parent := mod.aapt.LoggingParent
1785 if expected.logging_parent != logging_parent {
1786 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1787 expected.logging_parent, logging_parent)
1788 }
1789
Liz Kammer1d5983b2020-05-19 19:15:37 +00001790 // Check the package renaming flag, if exists.
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001791 res := variant.Output("package-res.apk")
1792 aapt2Flags := res.Args["flags"]
Liz Kammer1d5983b2020-05-19 19:15:37 +00001793 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1794 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1795 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001796 }
1797}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001798
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001799func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001800 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001801 android_app {
1802 name: "foo",
1803 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001804 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001805 }
1806
1807 override_android_app {
1808 name: "bar",
1809 base: "foo",
1810 package_name: "org.dandroid.bp",
1811 }
1812
1813 android_test {
1814 name: "baz",
1815 srcs: ["b.java"],
1816 instrumentation_for: "foo",
1817 }
1818
1819 android_test {
1820 name: "qux",
1821 srcs: ["b.java"],
1822 instrumentation_for: "bar",
1823 }
1824 `)
1825
1826 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1827 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1828 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1829 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1830 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1831 }
1832
1833 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1834 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1835 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1836 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1837 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1838 }
1839}
1840
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001841func TestOverrideAndroidTest(t *testing.T) {
1842 ctx, _ := testJava(t, `
1843 android_app {
1844 name: "foo",
1845 srcs: ["a.java"],
1846 package_name: "com.android.foo",
1847 sdk_version: "current",
1848 }
1849
1850 override_android_app {
1851 name: "bar",
1852 base: "foo",
1853 package_name: "com.android.bar",
1854 }
1855
1856 android_test {
1857 name: "foo_test",
1858 srcs: ["b.java"],
1859 instrumentation_for: "foo",
1860 }
1861
1862 override_android_test {
1863 name: "bar_test",
1864 base: "foo_test",
1865 package_name: "com.android.bar.test",
1866 instrumentation_for: "bar",
1867 instrumentation_target_package: "com.android.bar",
1868 }
1869 `)
1870
1871 expectedVariants := []struct {
1872 moduleName string
1873 variantName string
1874 apkPath string
1875 overrides []string
1876 targetVariant string
1877 packageFlag string
1878 targetPackageFlag string
1879 }{
1880 {
1881 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001882 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001883 overrides: nil,
1884 targetVariant: "android_common",
1885 packageFlag: "",
1886 targetPackageFlag: "",
1887 },
1888 {
1889 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001890 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001891 overrides: []string{"foo_test"},
1892 targetVariant: "android_common_bar",
1893 packageFlag: "com.android.bar.test",
1894 targetPackageFlag: "com.android.bar",
1895 },
1896 }
1897 for _, expected := range expectedVariants {
1898 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1899
1900 // Check the final apk name
1901 outputs := variant.AllOutputs()
1902 expectedApkPath := buildDir + expected.apkPath
1903 found := false
1904 for _, o := range outputs {
1905 if o == expectedApkPath {
1906 found = true
1907 break
1908 }
1909 }
1910 if !found {
1911 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1912 }
1913
1914 // Check if the overrides field values are correctly aggregated.
1915 mod := variant.Module().(*AndroidTest)
1916 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1917 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1918 expected.overrides, mod.appProperties.Overrides)
1919 }
1920
1921 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1922 javac := variant.Rule("javac")
1923 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1924 if !strings.Contains(javac.Args["classpath"], turbine) {
1925 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1926 }
1927
1928 // Check aapt2 flags.
1929 res := variant.Output("package-res.apk")
1930 aapt2Flags := res.Args["flags"]
1931 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1932 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1933 }
1934}
1935
Jaewoong Jung39982342020-01-14 10:27:18 -08001936func TestAndroidTest_FixTestConfig(t *testing.T) {
1937 ctx, _ := testJava(t, `
1938 android_app {
1939 name: "foo",
1940 srcs: ["a.java"],
1941 package_name: "com.android.foo",
1942 sdk_version: "current",
1943 }
1944
1945 android_test {
1946 name: "foo_test",
1947 srcs: ["b.java"],
1948 instrumentation_for: "foo",
1949 }
1950
1951 android_test {
1952 name: "bar_test",
1953 srcs: ["b.java"],
1954 package_name: "com.android.bar.test",
1955 instrumentation_for: "foo",
1956 }
1957
1958 override_android_test {
1959 name: "baz_test",
1960 base: "foo_test",
1961 package_name: "com.android.baz.test",
1962 }
1963 `)
1964
1965 testCases := []struct {
1966 moduleName string
1967 variantName string
1968 expectedFlags []string
1969 }{
1970 {
1971 moduleName: "foo_test",
1972 variantName: "android_common",
1973 },
1974 {
1975 moduleName: "bar_test",
1976 variantName: "android_common",
1977 expectedFlags: []string{
1978 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
1979 "--package-name com.android.bar.test",
1980 },
1981 },
1982 {
1983 moduleName: "foo_test",
1984 variantName: "android_common_baz_test",
1985 expectedFlags: []string{
1986 "--manifest " + buildDir +
1987 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
1988 "--package-name com.android.baz.test",
1989 "--test-file-name baz_test.apk",
1990 },
1991 },
1992 }
1993
1994 for _, test := range testCases {
1995 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
1996 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
1997
1998 if len(test.expectedFlags) > 0 {
1999 if params.Rule == nil {
2000 t.Errorf("test_config_fixer was expected to run, but didn't")
2001 } else {
2002 for _, flag := range test.expectedFlags {
2003 if !strings.Contains(params.RuleParams.Command, flag) {
2004 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2005 }
2006 }
2007 }
2008 } else {
2009 if params.Rule != nil {
2010 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2011 }
2012 }
2013
2014 }
2015}
2016
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002017func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002018 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002019 android_app_import {
2020 name: "foo",
2021 apk: "prebuilts/apk/app.apk",
2022 certificate: "platform",
2023 dex_preopt: {
2024 enabled: true,
2025 },
2026 }
2027 `)
2028
2029 variant := ctx.ModuleForTests("foo", "android_common")
2030
2031 // Check dexpreopt outputs.
2032 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2033 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2034 t.Errorf("can't find dexpreopt outputs")
2035 }
2036
2037 // Check cert signing flag.
2038 signedApk := variant.Output("signed/foo.apk")
2039 signingFlag := signedApk.Args["certificates"]
2040 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2041 if expected != signingFlag {
2042 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2043 }
2044}
2045
2046func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002047 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002048 android_app_import {
2049 name: "foo",
2050 apk: "prebuilts/apk/app.apk",
2051 certificate: "platform",
2052 dex_preopt: {
2053 enabled: false,
2054 },
2055 }
2056 `)
2057
2058 variant := ctx.ModuleForTests("foo", "android_common")
2059
2060 // Check dexpreopt outputs. They shouldn't exist.
2061 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2062 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2063 t.Errorf("dexpreopt shouldn't have run.")
2064 }
2065}
2066
2067func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002068 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002069 android_app_import {
2070 name: "foo",
2071 apk: "prebuilts/apk/app.apk",
2072 presigned: true,
2073 dex_preopt: {
2074 enabled: true,
2075 },
2076 }
2077 `)
2078
2079 variant := ctx.ModuleForTests("foo", "android_common")
2080
2081 // Check dexpreopt outputs.
2082 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2083 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2084 t.Errorf("can't find dexpreopt outputs")
2085 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002086 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002087 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2088 t.Errorf("signing rule shouldn't be included.")
2089 }
2090 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2091 t.Errorf("can't find aligning rule")
2092 }
2093}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002094
Liz Kammer24978992020-05-13 15:49:21 -07002095func TestAndroidAppImport_SigningLineage(t *testing.T) {
2096 ctx, _ := testJava(t, `
2097 android_app_import {
2098 name: "foo",
2099 apk: "prebuilts/apk/app.apk",
2100 certificate: "platform",
2101 lineage: "lineage.bin",
2102 }
2103 `)
2104
2105 variant := ctx.ModuleForTests("foo", "android_common")
2106
2107 // Check cert signing lineage flag.
2108 signedApk := variant.Output("signed/foo.apk")
2109 signingFlag := signedApk.Args["flags"]
2110 expected := "--lineage lineage.bin"
2111 if expected != signingFlag {
2112 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2113 }
2114}
2115
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002116func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2117 ctx, _ := testJava(t, `
2118 android_app_import {
2119 name: "foo",
2120 apk: "prebuilts/apk/app.apk",
2121 default_dev_cert: true,
2122 dex_preopt: {
2123 enabled: true,
2124 },
2125 }
2126 `)
2127
2128 variant := ctx.ModuleForTests("foo", "android_common")
2129
2130 // Check dexpreopt outputs.
2131 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2132 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2133 t.Errorf("can't find dexpreopt outputs")
2134 }
2135
2136 // Check cert signing flag.
2137 signedApk := variant.Output("signed/foo.apk")
2138 signingFlag := signedApk.Args["certificates"]
2139 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2140 if expected != signingFlag {
2141 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2142 }
2143}
2144
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002145func TestAndroidAppImport_DpiVariants(t *testing.T) {
2146 bp := `
2147 android_app_import {
2148 name: "foo",
2149 apk: "prebuilts/apk/app.apk",
2150 dpi_variants: {
2151 xhdpi: {
2152 apk: "prebuilts/apk/app_xhdpi.apk",
2153 },
2154 xxhdpi: {
2155 apk: "prebuilts/apk/app_xxhdpi.apk",
2156 },
2157 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002158 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002159 dex_preopt: {
2160 enabled: true,
2161 },
2162 }
2163 `
2164 testCases := []struct {
2165 name string
2166 aaptPreferredConfig *string
2167 aaptPrebuiltDPI []string
2168 expected string
2169 }{
2170 {
2171 name: "no preferred",
2172 aaptPreferredConfig: nil,
2173 aaptPrebuiltDPI: []string{},
2174 expected: "prebuilts/apk/app.apk",
2175 },
2176 {
2177 name: "AAPTPreferredConfig matches",
2178 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002179 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002180 expected: "prebuilts/apk/app_xhdpi.apk",
2181 },
2182 {
2183 name: "AAPTPrebuiltDPI matches",
2184 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2185 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2186 expected: "prebuilts/apk/app_xxhdpi.apk",
2187 },
2188 {
2189 name: "non-first AAPTPrebuiltDPI matches",
2190 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2191 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2192 expected: "prebuilts/apk/app_xhdpi.apk",
2193 },
2194 {
2195 name: "no matches",
2196 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2197 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2198 expected: "prebuilts/apk/app.apk",
2199 },
2200 }
2201
2202 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2203 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002204 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002205 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2206 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002207 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002208
2209 run(t, ctx, config)
2210
2211 variant := ctx.ModuleForTests("foo", "android_common")
2212 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2213 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2214 if len(matches) != 2 {
2215 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2216 }
2217 if test.expected != matches[1] {
2218 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2219 }
2220 }
2221}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002222
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002223func TestAndroidAppImport_Filename(t *testing.T) {
2224 ctx, config := testJava(t, `
2225 android_app_import {
2226 name: "foo",
2227 apk: "prebuilts/apk/app.apk",
2228 presigned: true,
2229 }
2230
2231 android_app_import {
2232 name: "bar",
2233 apk: "prebuilts/apk/app.apk",
2234 presigned: true,
2235 filename: "bar_sample.apk"
2236 }
2237 `)
2238
2239 testCases := []struct {
2240 name string
2241 expected string
2242 }{
2243 {
2244 name: "foo",
2245 expected: "foo.apk",
2246 },
2247 {
2248 name: "bar",
2249 expected: "bar_sample.apk",
2250 },
2251 }
2252
2253 for _, test := range testCases {
2254 variant := ctx.ModuleForTests(test.name, "android_common")
2255 if variant.MaybeOutput(test.expected).Rule == nil {
2256 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2257 }
2258
2259 a := variant.Module().(*AndroidAppImport)
2260 expectedValues := []string{test.expected}
2261 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002262 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002263 if !reflect.DeepEqual(actualValues, expectedValues) {
2264 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2265 actualValues, expectedValues)
2266 }
2267 }
2268}
2269
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002270func TestAndroidAppImport_ArchVariants(t *testing.T) {
2271 // The test config's target arch is ARM64.
2272 testCases := []struct {
2273 name string
2274 bp string
2275 expected string
2276 }{
2277 {
2278 name: "matching arch",
2279 bp: `
2280 android_app_import {
2281 name: "foo",
2282 apk: "prebuilts/apk/app.apk",
2283 arch: {
2284 arm64: {
2285 apk: "prebuilts/apk/app_arm64.apk",
2286 },
2287 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002288 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002289 dex_preopt: {
2290 enabled: true,
2291 },
2292 }
2293 `,
2294 expected: "prebuilts/apk/app_arm64.apk",
2295 },
2296 {
2297 name: "no matching arch",
2298 bp: `
2299 android_app_import {
2300 name: "foo",
2301 apk: "prebuilts/apk/app.apk",
2302 arch: {
2303 arm: {
2304 apk: "prebuilts/apk/app_arm.apk",
2305 },
2306 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002307 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002308 dex_preopt: {
2309 enabled: true,
2310 },
2311 }
2312 `,
2313 expected: "prebuilts/apk/app.apk",
2314 },
2315 }
2316
2317 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2318 for _, test := range testCases {
2319 ctx, _ := testJava(t, test.bp)
2320
2321 variant := ctx.ModuleForTests("foo", "android_common")
2322 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2323 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2324 if len(matches) != 2 {
2325 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2326 }
2327 if test.expected != matches[1] {
2328 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2329 }
2330 }
2331}
2332
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002333func TestAndroidTestImport(t *testing.T) {
2334 ctx, config := testJava(t, `
2335 android_test_import {
2336 name: "foo",
2337 apk: "prebuilts/apk/app.apk",
2338 presigned: true,
2339 data: [
2340 "testdata/data",
2341 ],
2342 }
2343 `)
2344
2345 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2346
2347 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002348 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002349 expected := []string{"tests"}
2350 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2351 if !reflect.DeepEqual(expected, actual) {
2352 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2353 }
2354 expected = []string{"testdata/data:testdata/data"}
2355 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2356 if !reflect.DeepEqual(expected, actual) {
2357 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2358 }
2359}
2360
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002361func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2362 ctx, _ := testJava(t, `
2363 android_test_import {
2364 name: "foo",
2365 apk: "prebuilts/apk/app.apk",
2366 certificate: "cert/new_cert",
2367 data: [
2368 "testdata/data",
2369 ],
2370 }
2371
2372 android_test_import {
2373 name: "foo_presigned",
2374 apk: "prebuilts/apk/app.apk",
2375 presigned: true,
2376 data: [
2377 "testdata/data",
2378 ],
2379 }
2380 `)
2381
2382 variant := ctx.ModuleForTests("foo", "android_common")
2383 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2384 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2385 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2386 }
2387
2388 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2389 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2390 if jniRule != android.Cp.String() {
2391 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2392 }
Liz Kammer3b70b3f2020-05-20 14:36:30 -07002393 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2394 t.Errorf("Presigned test apk should be aligned")
2395 }
2396}
2397
2398func TestAndroidTestImport_Preprocessed(t *testing.T) {
2399 ctx, _ := testJava(t, `
2400 android_test_import {
2401 name: "foo",
2402 apk: "prebuilts/apk/app.apk",
2403 presigned: true,
2404 preprocessed: true,
2405 }
2406
2407 android_test_import {
2408 name: "foo_cert",
2409 apk: "prebuilts/apk/app.apk",
2410 certificate: "cert/new_cert",
2411 preprocessed: true,
2412 }
2413 `)
2414
2415 testModules := []string{"foo", "foo_cert"}
2416 for _, m := range testModules {
2417 apkName := m + ".apk"
2418 variant := ctx.ModuleForTests(m, "android_common")
2419 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2420 if jniRule != android.Cp.String() {
2421 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2422 }
2423
2424 // Make sure signing and aligning were skipped.
2425 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2426 t.Errorf("signing rule shouldn't be included for preprocessed.")
2427 }
2428 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2429 t.Errorf("aligning rule shouldn't be for preprocessed")
2430 }
2431 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002432}
2433
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002434func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002435 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002436 cc_library {
2437 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002438 sdk_version: "current",
2439 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002440 }
2441
2442 android_test {
2443 name: "stl",
2444 jni_libs: ["libjni"],
2445 compile_multilib: "both",
2446 sdk_version: "current",
2447 stl: "c++_shared",
2448 }
2449
2450 android_test {
2451 name: "system",
2452 jni_libs: ["libjni"],
2453 compile_multilib: "both",
2454 sdk_version: "current",
2455 }
2456 `)
2457
2458 testCases := []struct {
2459 name string
2460 jnis []string
2461 }{
2462 {"stl",
2463 []string{
2464 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002465 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002466 },
2467 },
2468 {"system",
2469 []string{
2470 "libjni.so",
2471 },
2472 },
2473 }
2474
2475 for _, test := range testCases {
2476 t.Run(test.name, func(t *testing.T) {
2477 app := ctx.ModuleForTests(test.name, "android_common")
2478 jniLibZip := app.Output("jnilibs.zip")
2479 var jnis []string
2480 args := strings.Fields(jniLibZip.Args["jarArgs"])
2481 for i := 0; i < len(args); i++ {
2482 if args[i] == "-f" {
2483 jnis = append(jnis, args[i+1])
2484 i += 1
2485 }
2486 }
2487 jnisJoined := strings.Join(jnis, " ")
2488 for _, jni := range test.jnis {
2489 if !strings.Contains(jnisJoined, jni) {
2490 t.Errorf("missing jni %q in %q", jni, jnis)
2491 }
2492 }
2493 })
2494 }
2495}
Colin Cross50ddcc42019-05-16 12:28:22 -07002496
2497func TestUsesLibraries(t *testing.T) {
2498 bp := `
2499 java_sdk_library {
2500 name: "foo",
2501 srcs: ["a.java"],
2502 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002503 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002504 }
2505
2506 java_sdk_library {
Paul Duffin859fe962020-05-15 10:20:31 +01002507 name: "qux",
2508 srcs: ["a.java"],
2509 api_packages: ["qux"],
2510 sdk_version: "current",
2511 }
2512
2513 java_sdk_library {
2514 name: "quuz",
2515 srcs: ["a.java"],
2516 api_packages: ["quuz"],
2517 sdk_version: "current",
2518 }
2519
2520 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002521 name: "bar",
2522 srcs: ["a.java"],
2523 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002524 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002525 }
2526
2527 android_app {
2528 name: "app",
2529 srcs: ["a.java"],
Paul Duffin859fe962020-05-15 10:20:31 +01002530 libs: ["qux", "quuz.stubs"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002531 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002532 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002533 optional_uses_libs: [
2534 "bar",
2535 "baz",
2536 ],
2537 }
2538
2539 android_app_import {
2540 name: "prebuilt",
2541 apk: "prebuilts/apk/app.apk",
2542 certificate: "platform",
2543 uses_libs: ["foo"],
2544 optional_uses_libs: [
2545 "bar",
2546 "baz",
2547 ],
2548 }
2549 `
2550
Colin Cross98be1bb2019-12-13 20:41:13 -08002551 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002552 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2553
Colin Cross98be1bb2019-12-13 20:41:13 -08002554 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002555
2556 run(t, ctx, config)
2557
2558 app := ctx.ModuleForTests("app", "android_common")
2559 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2560
Paul Duffin859fe962020-05-15 10:20:31 +01002561 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2562 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2563 if w := "--uses-library qux"; !strings.Contains(manifestFixerArgs, w) {
2564 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2565 }
2566 if w := "--uses-library quuz"; !strings.Contains(manifestFixerArgs, w) {
2567 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2568 }
2569
Colin Cross50ddcc42019-05-16 12:28:22 -07002570 // Test that all libraries are verified
2571 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2572 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2573 t.Errorf("wanted %q in %q", w, cmd)
2574 }
2575
2576 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2577 t.Errorf("wanted %q in %q", w, cmd)
2578 }
2579
2580 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2581
2582 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2583 t.Errorf("wanted %q in %q", w, cmd)
2584 }
2585
2586 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2587 t.Errorf("wanted %q in %q", w, cmd)
2588 }
2589
2590 // Test that only present libraries are preopted
2591 cmd = app.Rule("dexpreopt").RuleParams.Command
2592
2593 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2594 t.Errorf("wanted %q in %q", w, cmd)
2595 }
2596
2597 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2598
2599 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2600 t.Errorf("wanted %q in %q", w, cmd)
2601 }
2602}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002603
2604func TestCodelessApp(t *testing.T) {
2605 testCases := []struct {
2606 name string
2607 bp string
2608 noCode bool
2609 }{
2610 {
2611 name: "normal",
2612 bp: `
2613 android_app {
2614 name: "foo",
2615 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002616 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002617 }
2618 `,
2619 noCode: false,
2620 },
2621 {
2622 name: "app without sources",
2623 bp: `
2624 android_app {
2625 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002626 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002627 }
2628 `,
2629 noCode: true,
2630 },
2631 {
2632 name: "app with libraries",
2633 bp: `
2634 android_app {
2635 name: "foo",
2636 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002637 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002638 }
2639
2640 java_library {
2641 name: "lib",
2642 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002643 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002644 }
2645 `,
2646 noCode: false,
2647 },
2648 {
2649 name: "app with sourceless libraries",
2650 bp: `
2651 android_app {
2652 name: "foo",
2653 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002654 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002655 }
2656
2657 java_library {
2658 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002659 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002660 }
2661 `,
2662 // TODO(jungjw): this should probably be true
2663 noCode: false,
2664 },
2665 }
2666
2667 for _, test := range testCases {
2668 t.Run(test.name, func(t *testing.T) {
2669 ctx := testApp(t, test.bp)
2670
2671 foo := ctx.ModuleForTests("foo", "android_common")
2672 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2673 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2674 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2675 }
2676 })
2677 }
2678}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002679
2680func TestEmbedNotice(t *testing.T) {
Colin Cross238c1f32020-06-07 16:58:18 -07002681 ctx, _ := testJavaWithFS(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002682 android_app {
2683 name: "foo",
2684 srcs: ["a.java"],
2685 static_libs: ["javalib"],
2686 jni_libs: ["libjni"],
2687 notice: "APP_NOTICE",
2688 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002689 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002690 }
2691
2692 // No embed_notice flag
2693 android_app {
2694 name: "bar",
2695 srcs: ["a.java"],
2696 jni_libs: ["libjni"],
2697 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002698 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002699 }
2700
2701 // No NOTICE files
2702 android_app {
2703 name: "baz",
2704 srcs: ["a.java"],
2705 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002706 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002707 }
2708
2709 cc_library {
2710 name: "libjni",
2711 system_shared_libs: [],
2712 stl: "none",
2713 notice: "LIB_NOTICE",
Colin Cross094cde42020-02-15 10:38:00 -08002714 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002715 }
2716
2717 java_library {
2718 name: "javalib",
2719 srcs: [
2720 ":gen",
2721 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002722 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002723 }
2724
2725 genrule {
2726 name: "gen",
2727 tools: ["gentool"],
2728 out: ["gen.java"],
2729 notice: "GENRULE_NOTICE",
2730 }
2731
2732 java_binary_host {
2733 name: "gentool",
2734 srcs: ["b.java"],
2735 notice: "TOOL_NOTICE",
2736 }
Colin Cross238c1f32020-06-07 16:58:18 -07002737 `, map[string][]byte{
2738 "APP_NOTICE": nil,
2739 "GENRULE_NOTICE": nil,
2740 "LIB_NOTICE": nil,
2741 "TOOL_NOTICE": nil,
2742 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002743
2744 // foo has NOTICE files to process, and embed_notices is true.
2745 foo := ctx.ModuleForTests("foo", "android_common")
2746 // verify merge notices rule.
2747 mergeNotices := foo.Rule("mergeNoticesRule")
2748 noticeInputs := mergeNotices.Inputs.Strings()
2749 // TOOL_NOTICE should be excluded as it's a host module.
2750 if len(mergeNotices.Inputs) != 3 {
2751 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2752 }
2753 if !inList("APP_NOTICE", noticeInputs) {
2754 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2755 }
2756 if !inList("LIB_NOTICE", noticeInputs) {
2757 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2758 }
2759 if !inList("GENRULE_NOTICE", noticeInputs) {
2760 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2761 }
2762 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2763 res := foo.Output("package-res.apk")
2764 aapt2Flags := res.Args["flags"]
2765 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2766 if !strings.Contains(aapt2Flags, e) {
2767 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2768 }
2769
2770 // bar has NOTICE files to process, but embed_notices is not set.
2771 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002772 res = bar.Output("package-res.apk")
2773 aapt2Flags = res.Args["flags"]
2774 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2775 if strings.Contains(aapt2Flags, e) {
2776 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002777 }
2778
2779 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2780 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002781 res = baz.Output("package-res.apk")
2782 aapt2Flags = res.Args["flags"]
2783 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2784 if strings.Contains(aapt2Flags, e) {
2785 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002786 }
2787}
Colin Cross53a87f52019-06-25 13:35:30 -07002788
2789func TestUncompressDex(t *testing.T) {
2790 testCases := []struct {
2791 name string
2792 bp string
2793
2794 uncompressedPlatform bool
2795 uncompressedUnbundled bool
2796 }{
2797 {
2798 name: "normal",
2799 bp: `
2800 android_app {
2801 name: "foo",
2802 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002803 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002804 }
2805 `,
2806 uncompressedPlatform: true,
2807 uncompressedUnbundled: false,
2808 },
2809 {
2810 name: "use_embedded_dex",
2811 bp: `
2812 android_app {
2813 name: "foo",
2814 use_embedded_dex: true,
2815 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002816 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002817 }
2818 `,
2819 uncompressedPlatform: true,
2820 uncompressedUnbundled: true,
2821 },
2822 {
2823 name: "privileged",
2824 bp: `
2825 android_app {
2826 name: "foo",
2827 privileged: true,
2828 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002829 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002830 }
2831 `,
2832 uncompressedPlatform: true,
2833 uncompressedUnbundled: true,
2834 },
David Srbeckye033cba2020-05-20 22:20:28 +01002835 {
2836 name: "normal_uncompress_dex_true",
2837 bp: `
2838 android_app {
2839 name: "foo",
2840 srcs: ["a.java"],
2841 sdk_version: "current",
2842 uncompress_dex: true,
2843 }
2844 `,
2845 uncompressedPlatform: true,
2846 uncompressedUnbundled: true,
2847 },
2848 {
2849 name: "normal_uncompress_dex_false",
2850 bp: `
2851 android_app {
2852 name: "foo",
2853 srcs: ["a.java"],
2854 sdk_version: "current",
2855 uncompress_dex: false,
2856 }
2857 `,
2858 uncompressedPlatform: false,
2859 uncompressedUnbundled: false,
2860 },
Colin Cross53a87f52019-06-25 13:35:30 -07002861 }
2862
2863 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2864 t.Helper()
2865
Colin Cross98be1bb2019-12-13 20:41:13 -08002866 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002867 if unbundled {
2868 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2869 }
2870
Colin Cross98be1bb2019-12-13 20:41:13 -08002871 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002872
2873 run(t, ctx, config)
2874
2875 foo := ctx.ModuleForTests("foo", "android_common")
2876 dex := foo.Rule("r8")
2877 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2878 aligned := foo.MaybeRule("zipalign").Rule != nil
2879
2880 if uncompressedInDexJar != want {
2881 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2882 }
2883
2884 if aligned != want {
2885 t.Errorf("want aligned %v, got %v", want, aligned)
2886 }
2887 }
2888
2889 for _, tt := range testCases {
2890 t.Run(tt.name, func(t *testing.T) {
2891 t.Run("platform", func(t *testing.T) {
2892 test(t, tt.bp, tt.uncompressedPlatform, false)
2893 })
2894 t.Run("unbundled", func(t *testing.T) {
2895 test(t, tt.bp, tt.uncompressedUnbundled, true)
2896 })
2897 })
2898 }
2899}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002900
2901func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2902 if expectedValue != "" {
2903 expectedFlag := "--" + flagName + " " + expectedValue
2904 if !strings.Contains(aapt2Flags, expectedFlag) {
2905 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2906 }
2907 } else {
2908 unexpectedFlag := "--" + flagName
2909 if strings.Contains(aapt2Flags, unexpectedFlag) {
2910 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2911 }
2912 }
2913}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002914
2915func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002916 fs := map[string][]byte{
2917 "baz/res/res/values/strings.xml": nil,
2918 "bar/res/res/values/strings.xml": nil,
2919 }
2920 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002921 runtime_resource_overlay {
2922 name: "foo",
2923 certificate: "platform",
Liz Kammer966b2f02020-05-19 16:15:25 -07002924 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002925 product_specific: true,
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002926 static_libs: ["bar"],
2927 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002928 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002929 }
2930
2931 runtime_resource_overlay {
2932 name: "foo_themed",
2933 certificate: "platform",
2934 product_specific: true,
2935 theme: "faza",
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002936 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002937 }
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002938
2939 android_library {
2940 name: "bar",
2941 resource_dirs: ["bar/res"],
2942 }
2943
2944 android_app {
2945 name: "baz",
2946 sdk_version: "current",
2947 resource_dirs: ["baz/res"],
2948 }
2949 `
2950 config := testAppConfig(nil, bp, fs)
2951 ctx := testContext()
2952 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002953
2954 m := ctx.ModuleForTests("foo", "android_common")
2955
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002956 // Check AAPT2 link flags.
2957 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2958 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2959 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2960 if len(absentFlags) > 0 {
2961 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2962 }
2963
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002964 // Check overlay.list output for static_libs dependency.
2965 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
2966 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
2967 if !inList(staticLibPackage, overlayList) {
2968 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
2969 }
2970
2971 // Check AAPT2 link flags for resource_libs dependency.
2972 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
2973 if !strings.Contains(aapt2Flags, resourceLibFlag) {
2974 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
2975 }
2976
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002977 // Check cert signing flag.
2978 signedApk := m.Output("signed/foo.apk")
Liz Kammer966b2f02020-05-19 16:15:25 -07002979 lineageFlag := signedApk.Args["flags"]
2980 expectedLineageFlag := "--lineage lineage.bin"
2981 if expectedLineageFlag != lineageFlag {
2982 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
2983 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002984 signingFlag := signedApk.Args["certificates"]
2985 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2986 if expected != signingFlag {
2987 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2988 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002989 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2990 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002991 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
2992 if !reflect.DeepEqual(path, expectedPath) {
2993 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
2994 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002995
2996 // Check device location.
Jaewoong Jungad0177b2020-04-24 15:22:40 -07002997 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002998 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002999 if !reflect.DeepEqual(path, expectedPath) {
3000 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3001 }
3002
3003 // A themed module has a different device location
3004 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003005 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3006 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003007 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3008 if !reflect.DeepEqual(path, expectedPath) {
3009 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3010 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003011
3012 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3013 expectedOverrides := []string{"foo"}
3014 if !reflect.DeepEqual(overrides, expectedOverrides) {
3015 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3016 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003017}
Roshan Pius4df2bc72020-04-27 09:42:27 -07003018
3019func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3020 ctx, _ := testJava(t, `
3021 runtime_resource_overlay {
3022 name: "foo_overlay",
3023 certificate: "platform",
3024 product_specific: true,
3025 sdk_version: "current",
3026 }
3027
3028 override_runtime_resource_overlay {
3029 name: "bar_overlay",
3030 base: "foo_overlay",
3031 package_name: "com.android.bar.overlay",
3032 target_package_name: "com.android.bar",
3033 }
3034 `)
3035
3036 expectedVariants := []struct {
3037 moduleName string
3038 variantName string
3039 apkPath string
3040 overrides []string
3041 targetVariant string
3042 packageFlag string
3043 targetPackageFlag string
3044 }{
3045 {
3046 variantName: "android_common",
3047 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3048 overrides: nil,
3049 targetVariant: "android_common",
3050 packageFlag: "",
3051 targetPackageFlag: "",
3052 },
3053 {
3054 variantName: "android_common_bar_overlay",
3055 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3056 overrides: []string{"foo_overlay"},
3057 targetVariant: "android_common_bar",
3058 packageFlag: "com.android.bar.overlay",
3059 targetPackageFlag: "com.android.bar",
3060 },
3061 }
3062 for _, expected := range expectedVariants {
3063 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3064
3065 // Check the final apk name
3066 outputs := variant.AllOutputs()
3067 expectedApkPath := buildDir + expected.apkPath
3068 found := false
3069 for _, o := range outputs {
3070 if o == expectedApkPath {
3071 found = true
3072 break
3073 }
3074 }
3075 if !found {
3076 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3077 }
3078
3079 // Check if the overrides field values are correctly aggregated.
3080 mod := variant.Module().(*RuntimeResourceOverlay)
3081 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3082 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3083 expected.overrides, mod.properties.Overrides)
3084 }
3085
3086 // Check aapt2 flags.
3087 res := variant.Output("package-res.apk")
3088 aapt2Flags := res.Args["flags"]
3089 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
3090 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3091 }
3092}
Jaewoong Jungbf135462020-04-26 15:10:51 -07003093
3094func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3095 ctx, config := testJava(t, `
3096 java_defaults {
3097 name: "rro_defaults",
3098 theme: "default_theme",
3099 product_specific: true,
3100 aaptflags: ["--keep-raw-values"],
3101 }
3102
3103 runtime_resource_overlay {
3104 name: "foo_with_defaults",
3105 defaults: ["rro_defaults"],
3106 }
3107
3108 runtime_resource_overlay {
3109 name: "foo_barebones",
3110 }
3111 `)
3112
3113 //
3114 // RRO module with defaults
3115 //
3116 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3117
3118 // Check AAPT2 link flags.
3119 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3120 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3121 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3122 if len(absentFlags) > 0 {
3123 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3124 }
3125
3126 // Check device location.
3127 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3128 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3129 if !reflect.DeepEqual(path, expectedPath) {
3130 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3131 }
3132
3133 //
3134 // RRO module without defaults
3135 //
3136 m = ctx.ModuleForTests("foo_barebones", "android_common")
3137
3138 // Check AAPT2 link flags.
3139 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3140 unexpectedFlags := "--keep-raw-values"
3141 if inList(unexpectedFlags, aapt2Flags) {
3142 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3143 }
3144
3145 // Check device location.
3146 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3147 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3148 if !reflect.DeepEqual(path, expectedPath) {
3149 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3150 }
3151}