blob: 620a8dd7537a245139272b13311c61a7a2b5f316 [file] [log] [blame]
Colin Cross3bc7ffa2017-11-22 16:19:37 -08001// Copyright 2017 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package java
16
17import (
Colin Crossd09b0b62018-04-18 11:06:47 -070018 "fmt"
Colin Crossa4f08812018-10-02 22:03:40 -070019 "path/filepath"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080020 "reflect"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070021 "regexp"
Colin Crossb69301e2017-12-01 10:48:26 -080022 "sort"
Colin Crossd09b0b62018-04-18 11:06:47 -070023 "strings"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080024 "testing"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070025
26 "github.com/google/blueprint/proptools"
27
28 "android/soong/android"
29 "android/soong/cc"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080030)
31
32var (
33 resourceFiles = []string{
34 "res/layout/layout.xml",
35 "res/values/strings.xml",
36 "res/values-en-rUS/strings.xml",
37 }
38
39 compiledResourceFiles = []string{
40 "aapt2/res/layout_layout.xml.flat",
41 "aapt2/res/values_strings.arsc.flat",
42 "aapt2/res/values-en-rUS_strings.arsc.flat",
43 }
44)
45
Colin Cross98be1bb2019-12-13 20:41:13 -080046func testAppConfig(env map[string]string, bp string, fs map[string][]byte) android.Config {
Colin Cross527012a2017-11-30 22:56:16 -080047 appFS := map[string][]byte{}
48 for k, v := range fs {
49 appFS[k] = v
Colin Cross3bc7ffa2017-11-22 16:19:37 -080050 }
51
Colin Cross527012a2017-11-30 22:56:16 -080052 for _, file := range resourceFiles {
53 appFS[file] = nil
54 }
55
Colin Cross98be1bb2019-12-13 20:41:13 -080056 return testConfig(env, bp, appFS)
Colin Cross527012a2017-11-30 22:56:16 -080057}
58
59func testApp(t *testing.T, bp string) *android.TestContext {
Colin Cross98be1bb2019-12-13 20:41:13 -080060 config := testAppConfig(nil, bp, nil)
Colin Cross527012a2017-11-30 22:56:16 -080061
Colin Cross98be1bb2019-12-13 20:41:13 -080062 ctx := testContext()
Colin Cross527012a2017-11-30 22:56:16 -080063
64 run(t, ctx, config)
65
66 return ctx
Colin Cross3bc7ffa2017-11-22 16:19:37 -080067}
68
69func TestApp(t *testing.T) {
Colin Crossa97c5d32018-03-28 14:58:31 -070070 for _, moduleType := range []string{"android_app", "android_library"} {
71 t.Run(moduleType, func(t *testing.T) {
72 ctx := testApp(t, moduleType+` {
73 name: "foo",
74 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +090075 sdk_version: "current"
Colin Crossa97c5d32018-03-28 14:58:31 -070076 }
77 `)
Colin Cross3bc7ffa2017-11-22 16:19:37 -080078
Colin Crossa97c5d32018-03-28 14:58:31 -070079 foo := ctx.ModuleForTests("foo", "android_common")
Colin Cross3bc7ffa2017-11-22 16:19:37 -080080
Colin Cross31656952018-05-24 16:11:20 -070081 var expectedLinkImplicits []string
82
83 manifestFixer := foo.Output("manifest_fixer/AndroidManifest.xml")
84 expectedLinkImplicits = append(expectedLinkImplicits, manifestFixer.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080085
Colin Crossa97c5d32018-03-28 14:58:31 -070086 frameworkRes := ctx.ModuleForTests("framework-res", "android_common")
87 expectedLinkImplicits = append(expectedLinkImplicits,
88 frameworkRes.Output("package-res.apk").Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080089
Colin Crossa97c5d32018-03-28 14:58:31 -070090 // Test the mapping from input files to compiled output file names
91 compile := foo.Output(compiledResourceFiles[0])
92 if !reflect.DeepEqual(resourceFiles, compile.Inputs.Strings()) {
93 t.Errorf("expected aapt2 compile inputs expected:\n %#v\n got:\n %#v",
94 resourceFiles, compile.Inputs.Strings())
95 }
Colin Crossb69301e2017-12-01 10:48:26 -080096
Colin Crossa97c5d32018-03-28 14:58:31 -070097 compiledResourceOutputs := compile.Outputs.Strings()
98 sort.Strings(compiledResourceOutputs)
Colin Crossb69301e2017-12-01 10:48:26 -080099
Colin Crossa97c5d32018-03-28 14:58:31 -0700100 expectedLinkImplicits = append(expectedLinkImplicits, compiledResourceOutputs...)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800101
Colin Crossa97c5d32018-03-28 14:58:31 -0700102 list := foo.Output("aapt2/res.list")
103 expectedLinkImplicits = append(expectedLinkImplicits, list.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800104
Colin Crossa97c5d32018-03-28 14:58:31 -0700105 // Check that the link rule uses
106 res := ctx.ModuleForTests("foo", "android_common").Output("package-res.apk")
107 if !reflect.DeepEqual(expectedLinkImplicits, res.Implicits.Strings()) {
108 t.Errorf("expected aapt2 link implicits expected:\n %#v\n got:\n %#v",
109 expectedLinkImplicits, res.Implicits.Strings())
110 }
111 })
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800112 }
113}
Colin Cross890ff552017-11-30 20:13:19 -0800114
Colin Crosse560c4a2019-03-19 16:03:11 -0700115func TestAppSplits(t *testing.T) {
116 ctx := testApp(t, `
117 android_app {
118 name: "foo",
119 srcs: ["a.java"],
120 package_splits: ["v4", "v7,hdpi"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900121 sdk_version: "current"
Colin Crosse560c4a2019-03-19 16:03:11 -0700122 }`)
123
124 foo := ctx.ModuleForTests("foo", "android_common")
125
126 expectedOutputs := []string{
127 filepath.Join(buildDir, ".intermediates/foo/android_common/foo.apk"),
128 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v4.apk"),
129 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v7_hdpi.apk"),
130 }
131 for _, expectedOutput := range expectedOutputs {
132 foo.Output(expectedOutput)
133 }
134
Colin Cross41955e82019-05-29 14:40:35 -0700135 outputFiles, err := foo.Module().(*AndroidApp).OutputFiles("")
136 if err != nil {
137 t.Fatal(err)
138 }
139 if g, w := outputFiles.Strings(), expectedOutputs; !reflect.DeepEqual(g, w) {
140 t.Errorf(`want OutputFiles("") = %q, got %q`, w, g)
Colin Crosse560c4a2019-03-19 16:03:11 -0700141 }
142}
143
Sasha Smundak4de27a52020-04-23 09:49:59 -0700144func TestAndroidAppSet(t *testing.T) {
145 ctx, config := testJava(t, `
146 android_app_set {
147 name: "foo",
148 set: "prebuilts/apks/app.apks",
149 prerelease: true,
Jaewoong Jung8bec0262020-06-29 19:18:44 -0700150 }`)
Sasha Smundak4de27a52020-04-23 09:49:59 -0700151 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700152 const packedSplitApks = "foo.zip"
Sasha Smundak4de27a52020-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 Jung8bec0262020-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 Smundak4de27a52020-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 {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700179 name string
180 targets []android.Target
181 aaptPrebuiltDPI []string
182 sdkVersion int
183 expected map[string]string
Sasha Smundak4de27a52020-04-23 09:49:59 -0700184 }{
185 {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700186 name: "One",
187 targets: []android.Target{
188 {Os: android.Android, Arch: android.Arch{ArchType: android.X86}},
189 },
Sasha Smundak4de27a52020-04-23 09:49:59 -0700190 aaptPrebuiltDPI: []string{"ldpi", "xxhdpi"},
191 sdkVersion: 29,
192 expected: map[string]string{
193 "abis": "X86",
194 "allow-prereleased": "false",
195 "screen-densities": "LDPI,XXHDPI",
196 "sdk-version": "29",
197 "stem": "foo",
198 },
199 },
200 {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700201 name: "Two",
202 targets: []android.Target{
203 {Os: android.Android, Arch: android.Arch{ArchType: android.X86_64}},
204 {Os: android.Android, Arch: android.Arch{ArchType: android.X86}},
205 },
206 aaptPrebuiltDPI: nil,
207 sdkVersion: 30,
Sasha Smundak4de27a52020-04-23 09:49:59 -0700208 expected: map[string]string{
209 "abis": "X86_64,X86",
210 "allow-prereleased": "false",
211 "screen-densities": "all",
212 "sdk-version": "30",
213 "stem": "foo",
214 },
215 },
216 }
217
218 for _, test := range testCases {
219 config := testAppConfig(nil, bp, nil)
220 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
221 config.TestProductVariables.Platform_sdk_version = &test.sdkVersion
Jaewoong Jung829b7132020-06-10 12:23:32 -0700222 config.Targets[android.Android] = test.targets
Sasha Smundak4de27a52020-04-23 09:49:59 -0700223 ctx := testContext()
224 run(t, ctx, config)
225 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700226 const packedSplitApks = "foo.zip"
Sasha Smundak4de27a52020-04-23 09:49:59 -0700227 params := module.Output(packedSplitApks)
228 for k, v := range test.expected {
229 if actual := params.Args[k]; actual != v {
230 t.Errorf("%s: bad build arg value for '%s': '%s', expected '%s'",
231 test.name, k, actual, v)
232 }
233 }
234 }
235}
236
Jeongik Cha538c0d02019-07-11 15:54:27 +0900237func TestPlatformAPIs(t *testing.T) {
238 testJava(t, `
239 android_app {
240 name: "foo",
241 srcs: ["a.java"],
242 platform_apis: true,
243 }
244 `)
245
246 testJava(t, `
247 android_app {
248 name: "foo",
249 srcs: ["a.java"],
250 sdk_version: "current",
251 }
252 `)
253
254 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
255 android_app {
256 name: "bar",
257 srcs: ["b.java"],
258 }
259 `)
260
261 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
262 android_app {
263 name: "bar",
264 srcs: ["b.java"],
265 sdk_version: "system_current",
266 platform_apis: true,
267 }
268 `)
269}
270
Jeongik Chae403e9e2019-12-07 00:16:24 +0900271func TestAndroidAppLinkType(t *testing.T) {
272 testJava(t, `
273 android_app {
274 name: "foo",
275 srcs: ["a.java"],
276 libs: ["bar"],
277 static_libs: ["baz"],
278 platform_apis: true,
279 }
280
281 java_library {
282 name: "bar",
283 sdk_version: "current",
284 srcs: ["b.java"],
285 }
286
287 android_library {
288 name: "baz",
289 sdk_version: "system_current",
290 srcs: ["c.java"],
291 }
292 `)
293
294 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.", `
295 android_app {
296 name: "foo",
297 srcs: ["a.java"],
298 libs: ["bar"],
299 sdk_version: "current",
300 static_libs: ["baz"],
301 }
302
303 java_library {
304 name: "bar",
305 sdk_version: "current",
306 srcs: ["b.java"],
307 }
308
309 android_library {
310 name: "baz",
311 sdk_version: "system_current",
312 srcs: ["c.java"],
313 }
314 `)
315
316 testJava(t, `
317 android_app {
318 name: "foo",
319 srcs: ["a.java"],
320 libs: ["bar"],
321 sdk_version: "system_current",
322 static_libs: ["baz"],
323 }
324
325 java_library {
326 name: "bar",
327 sdk_version: "current",
328 srcs: ["b.java"],
329 }
330
331 android_library {
332 name: "baz",
333 sdk_version: "system_current",
334 srcs: ["c.java"],
335 }
336 `)
337
338 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.", `
339 android_app {
340 name: "foo",
341 srcs: ["a.java"],
342 libs: ["bar"],
343 sdk_version: "system_current",
344 static_libs: ["baz"],
345 }
346
347 java_library {
348 name: "bar",
349 sdk_version: "current",
350 srcs: ["b.java"],
351 }
352
353 android_library {
354 name: "baz",
355 srcs: ["c.java"],
356 }
357 `)
358}
359
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100360func TestUpdatableApps(t *testing.T) {
361 testCases := []struct {
362 name string
363 bp string
364 expectedError string
365 }{
366 {
367 name: "Stable public SDK",
368 bp: `android_app {
369 name: "foo",
370 srcs: ["a.java"],
371 sdk_version: "29",
Artur Satayev11962102020-04-16 13:43:02 +0100372 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100373 updatable: true,
374 }`,
375 },
376 {
377 name: "Stable system SDK",
378 bp: `android_app {
379 name: "foo",
380 srcs: ["a.java"],
381 sdk_version: "system_29",
Artur Satayev11962102020-04-16 13:43:02 +0100382 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100383 updatable: true,
384 }`,
385 },
386 {
387 name: "Current public SDK",
388 bp: `android_app {
389 name: "foo",
390 srcs: ["a.java"],
391 sdk_version: "current",
Artur Satayev11962102020-04-16 13:43:02 +0100392 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100393 updatable: true,
394 }`,
395 },
396 {
397 name: "Current system SDK",
398 bp: `android_app {
399 name: "foo",
400 srcs: ["a.java"],
401 sdk_version: "system_current",
Artur Satayev11962102020-04-16 13:43:02 +0100402 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100403 updatable: true,
404 }`,
405 },
406 {
407 name: "Current module SDK",
408 bp: `android_app {
409 name: "foo",
410 srcs: ["a.java"],
411 sdk_version: "module_current",
Artur Satayev11962102020-04-16 13:43:02 +0100412 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100413 updatable: true,
414 }`,
415 },
416 {
417 name: "Current core SDK",
418 bp: `android_app {
419 name: "foo",
420 srcs: ["a.java"],
421 sdk_version: "core_current",
Artur Satayev11962102020-04-16 13:43:02 +0100422 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100423 updatable: true,
424 }`,
425 },
426 {
427 name: "No Platform APIs",
428 bp: `android_app {
429 name: "foo",
430 srcs: ["a.java"],
431 platform_apis: true,
Artur Satayev11962102020-04-16 13:43:02 +0100432 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100433 updatable: true,
434 }`,
435 expectedError: "Updatable apps must use stable SDKs",
436 },
437 {
438 name: "No Core Platform APIs",
439 bp: `android_app {
440 name: "foo",
441 srcs: ["a.java"],
442 sdk_version: "core_platform",
Artur Satayev11962102020-04-16 13:43:02 +0100443 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100444 updatable: true,
445 }`,
446 expectedError: "Updatable apps must use stable SDKs",
447 },
448 {
449 name: "No unspecified APIs",
450 bp: `android_app {
451 name: "foo",
452 srcs: ["a.java"],
453 updatable: true,
Artur Satayev11962102020-04-16 13:43:02 +0100454 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100455 }`,
456 expectedError: "Updatable apps must use stable SDK",
457 },
Artur Satayev11962102020-04-16 13:43:02 +0100458 {
459 name: "Must specify min_sdk_version",
460 bp: `android_app {
461 name: "app_without_min_sdk_version",
462 srcs: ["a.java"],
463 sdk_version: "29",
464 updatable: true,
465 }`,
466 expectedError: "updatable apps must set min_sdk_version.",
467 },
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100468 }
469
470 for _, test := range testCases {
471 t.Run(test.name, func(t *testing.T) {
472 if test.expectedError == "" {
473 testJava(t, test.bp)
474 } else {
475 testJavaError(t, test.expectedError, test.bp)
476 }
477 })
478 }
479}
480
Jooyung Hanaf7f91f2020-04-29 14:01:06 +0900481func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
482 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
483 android_app {
484 name: "foo",
485 srcs: ["a.java"],
486 updatable: true,
487 sdk_version: "current",
488 min_sdk_version: "current",
489 jni_libs: ["libjni"],
490 }
491
492 cc_library {
493 name: "libjni",
494 stl: "none",
495 system_shared_libs: [],
496 sdk_version: "current",
497 }
498 `)
499}
500
501func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
502 bp := cc.GatherRequiredDepsForTest(android.Android) + `
503 android_app {
504 name: "foo",
505 srcs: ["a.java"],
506 updatable: true,
507 sdk_version: "current",
508 min_sdk_version: "29",
509 jni_libs: ["libjni"],
510 }
511
512 cc_library {
513 name: "libjni",
514 stl: "none",
515 system_shared_libs: [],
516 sdk_version: "29",
517 }
518
519 ndk_prebuilt_object {
520 name: "ndk_crtbegin_so.29",
521 sdk_version: "29",
522 }
523
524 ndk_prebuilt_object {
525 name: "ndk_crtend_so.29",
526 sdk_version: "29",
527 }
528 `
529 fs := map[string][]byte{
530 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
531 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
532 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
533 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
534 }
535
536 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
537
538 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
539 var crtbeginFound, crtendFound bool
540 for _, input := range inputs {
541 switch input.String() {
542 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o":
543 crtbeginFound = true
544 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o":
545 crtendFound = true
546 }
547 }
548 if !crtbeginFound || !crtendFound {
549 t.Error("should link with ndk_crtbegin_so.29 and ndk_crtend_so.29")
550 }
551}
552
553func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
554 bp := cc.GatherRequiredDepsForTest(android.Android) + `
555 android_app {
556 name: "foo",
557 srcs: ["a.java"],
558 updatable: true,
559 sdk_version: "current",
560 min_sdk_version: "29", // this APK should support 29
561 jni_libs: ["libjni"],
562 }
563
564 cc_library {
565 name: "libjni",
566 stl: "none",
567 sdk_version: "current",
568 }
569 `
570 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
571}
572
573func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
574 bp := cc.GatherRequiredDepsForTest(android.Android) + `
575 android_app {
576 name: "foo",
577 srcs: ["a.java"],
578 updatable: true,
579 sdk_version: "current",
580 min_sdk_version: "29", // this APK should support 29
581 jni_libs: ["libjni"],
582 }
583
584 cc_library {
585 name: "libjni",
586 stl: "none",
587 shared_libs: ["libbar"],
588 system_shared_libs: [],
589 sdk_version: "27",
590 }
591
592 cc_library {
593 name: "libbar",
594 stl: "none",
595 system_shared_libs: [],
596 sdk_version: "current",
597 }
598 `
599 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
600}
601
Colin Cross0ddae7f2019-02-07 15:30:01 -0800602func TestResourceDirs(t *testing.T) {
603 testCases := []struct {
604 name string
605 prop string
606 resources []string
607 }{
608 {
609 name: "no resource_dirs",
610 prop: "",
611 resources: []string{"res/res/values/strings.xml"},
612 },
613 {
614 name: "resource_dirs",
615 prop: `resource_dirs: ["res"]`,
616 resources: []string{"res/res/values/strings.xml"},
617 },
618 {
619 name: "empty resource_dirs",
620 prop: `resource_dirs: []`,
621 resources: nil,
622 },
623 }
624
625 fs := map[string][]byte{
626 "res/res/values/strings.xml": nil,
627 }
628
629 bp := `
630 android_app {
631 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900632 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800633 %s
634 }
635 `
636
637 for _, testCase := range testCases {
638 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800639 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
640 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800641 run(t, ctx, config)
642
643 module := ctx.ModuleForTests("foo", "android_common")
644 resourceList := module.MaybeOutput("aapt2/res.list")
645
646 var resources []string
647 if resourceList.Rule != nil {
648 for _, compiledResource := range resourceList.Inputs.Strings() {
649 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
650 }
651 }
652
653 if !reflect.DeepEqual(resources, testCase.resources) {
654 t.Errorf("expected resource files %q, got %q",
655 testCase.resources, resources)
656 }
657 })
658 }
659}
660
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800661func TestLibraryAssets(t *testing.T) {
662 bp := `
663 android_app {
664 name: "foo",
665 sdk_version: "current",
666 static_libs: ["lib1", "lib2", "lib3"],
667 }
668
669 android_library {
670 name: "lib1",
671 sdk_version: "current",
672 asset_dirs: ["assets_a"],
673 }
674
675 android_library {
676 name: "lib2",
677 sdk_version: "current",
678 }
679
680 android_library {
681 name: "lib3",
682 sdk_version: "current",
683 static_libs: ["lib4"],
684 }
685
686 android_library {
687 name: "lib4",
688 sdk_version: "current",
689 asset_dirs: ["assets_b"],
690 }
691 `
692
693 testCases := []struct {
694 name string
695 assetFlag string
696 assetPackages []string
697 }{
698 {
699 name: "foo",
700 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
701 assetPackages: []string{
702 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
703 buildDir + "/.intermediates/lib1/android_common/assets.zip",
704 buildDir + "/.intermediates/lib3/android_common/assets.zip",
705 },
706 },
707 {
708 name: "lib1",
709 assetFlag: "-A assets_a",
710 },
711 {
712 name: "lib2",
713 },
714 {
715 name: "lib3",
716 assetPackages: []string{
717 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
718 buildDir + "/.intermediates/lib4/android_common/assets.zip",
719 },
720 },
721 {
722 name: "lib4",
723 assetFlag: "-A assets_b",
724 },
725 }
726 ctx := testApp(t, bp)
727
728 for _, test := range testCases {
729 t.Run(test.name, func(t *testing.T) {
730 m := ctx.ModuleForTests(test.name, "android_common")
731
732 // Check asset flag in aapt2 link flags
733 var aapt2link android.TestingBuildParams
734 if len(test.assetPackages) > 0 {
735 aapt2link = m.Output("aapt2/package-res.apk")
736 } else {
737 aapt2link = m.Output("package-res.apk")
738 }
739 aapt2Flags := aapt2link.Args["flags"]
740 if test.assetFlag != "" {
741 if !strings.Contains(aapt2Flags, test.assetFlag) {
742 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
743 }
744 } else {
745 if strings.Contains(aapt2Flags, " -A ") {
746 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
747 }
748 }
749
750 // Check asset merge rule.
751 if len(test.assetPackages) > 0 {
752 mergeAssets := m.Output("package-res.apk")
753 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
754 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
755 mergeAssets.Inputs.Strings(), test.assetPackages)
756 }
757 }
758 })
759 }
760}
761
Colin Crossbec85302019-02-13 13:15:46 -0800762func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800763 testCases := []struct {
764 name string
765 enforceRROTargets []string
766 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800767 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800768 overlayFiles map[string][]string
769 rroDirs map[string][]string
770 }{
771 {
772 name: "no RRO",
773 enforceRROTargets: nil,
774 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800775 resourceFiles: map[string][]string{
776 "foo": nil,
777 "bar": {"bar/res/res/values/strings.xml"},
778 "lib": nil,
779 "lib2": {"lib2/res/res/values/strings.xml"},
780 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800781 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800782 "foo": {
783 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800784 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000785 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800786 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800787 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
788 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000789 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800790 },
Colin Crossbec85302019-02-13 13:15:46 -0800791 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800792 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
793 "device/vendor/blah/overlay/bar/res/values/strings.xml",
794 },
Colin Crossbec85302019-02-13 13:15:46 -0800795 "lib": {
796 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
797 "lib/res/res/values/strings.xml",
798 "device/vendor/blah/overlay/lib/res/values/strings.xml",
799 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800800 },
801 rroDirs: map[string][]string{
802 "foo": nil,
803 "bar": nil,
804 },
805 },
806 {
807 name: "enforce RRO on foo",
808 enforceRROTargets: []string{"foo"},
809 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800810 resourceFiles: map[string][]string{
811 "foo": nil,
812 "bar": {"bar/res/res/values/strings.xml"},
813 "lib": nil,
814 "lib2": {"lib2/res/res/values/strings.xml"},
815 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800816 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800817 "foo": {
818 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800819 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000820 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800821 "foo/res/res/values/strings.xml",
822 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
823 },
Colin Crossbec85302019-02-13 13:15:46 -0800824 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800825 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
826 "device/vendor/blah/overlay/bar/res/values/strings.xml",
827 },
Colin Crossbec85302019-02-13 13:15:46 -0800828 "lib": {
829 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
830 "lib/res/res/values/strings.xml",
831 "device/vendor/blah/overlay/lib/res/values/strings.xml",
832 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800833 },
Colin Crossc1c37552019-01-31 11:42:41 -0800834
Colin Cross5c4791c2019-02-01 11:44:44 -0800835 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800836 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000837 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800838 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
839 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000840 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800841 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800842 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800843 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800844 },
845 },
846 {
847 name: "enforce RRO on all",
848 enforceRROTargets: []string{"*"},
849 enforceRROExcludedOverlays: []string{
850 // Excluding specific apps/res directories also allowed.
851 "device/vendor/blah/static_overlay/foo",
852 "device/vendor/blah/static_overlay/bar/res",
853 },
Colin Crossbec85302019-02-13 13:15:46 -0800854 resourceFiles: map[string][]string{
855 "foo": nil,
856 "bar": {"bar/res/res/values/strings.xml"},
857 "lib": nil,
858 "lib2": {"lib2/res/res/values/strings.xml"},
859 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800860 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800861 "foo": {
862 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800863 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000864 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800865 "foo/res/res/values/strings.xml",
866 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
867 },
Colin Crossbec85302019-02-13 13:15:46 -0800868 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
869 "lib": {
870 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
871 "lib/res/res/values/strings.xml",
872 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800873 },
874 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800875 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000876 "device:device/vendor/blah/overlay/foo/res",
877 "product:product/vendor/blah/overlay/foo/res",
878 // Lib dep comes after the direct deps
879 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800880 },
Anton Hansson53c88442019-03-18 15:53:16 +0000881 "bar": {"device:device/vendor/blah/overlay/bar/res"},
882 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800883 },
884 },
885 }
886
Anton Hansson53c88442019-03-18 15:53:16 +0000887 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800888 "device/vendor/blah/overlay",
889 "device/vendor/blah/overlay2",
890 "device/vendor/blah/static_overlay",
891 }
892
Anton Hansson53c88442019-03-18 15:53:16 +0000893 productResourceOverlays := []string{
894 "product/vendor/blah/overlay",
895 }
896
Colin Cross890ff552017-11-30 20:13:19 -0800897 fs := map[string][]byte{
898 "foo/res/res/values/strings.xml": nil,
899 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800900 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800901 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800902 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
903 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800904 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800905 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
906 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
907 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000908 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800909 }
910
911 bp := `
912 android_app {
913 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900914 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800915 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000916 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800917 }
918
919 android_app {
920 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900921 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800922 resource_dirs: ["bar/res"],
923 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800924
925 android_library {
926 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900927 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800928 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800929 static_libs: ["lib2"],
930 }
931
932 android_library {
933 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900934 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800935 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800936 }
Anton Hansson53c88442019-03-18 15:53:16 +0000937
938 // This library has the same resources as lib (should not lead to dupe RROs)
939 android_library {
940 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900941 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000942 resource_dirs: ["lib/res"]
943 }
Colin Cross890ff552017-11-30 20:13:19 -0800944 `
945
Colin Cross5c4791c2019-02-01 11:44:44 -0800946 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800947 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800948 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000949 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
950 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800951 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800952 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800953 }
954 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800955 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800956 }
957
Colin Cross98be1bb2019-12-13 20:41:13 -0800958 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800959 run(t, ctx, config)
960
Colin Crossbec85302019-02-13 13:15:46 -0800961 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
962 for _, o := range list {
963 res := module.MaybeOutput(o)
964 if res.Rule != nil {
965 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
966 // verify the inputs to the .arsc.flat rule.
967 files = append(files, res.Inputs.Strings()...)
968 } else {
969 // Otherwise, verify the full path to the output of the other module
970 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000971 }
Colin Cross890ff552017-11-30 20:13:19 -0800972 }
Colin Crossbec85302019-02-13 13:15:46 -0800973 return files
Colin Cross890ff552017-11-30 20:13:19 -0800974 }
975
Colin Crossbec85302019-02-13 13:15:46 -0800976 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
977 module := ctx.ModuleForTests(moduleName, "android_common")
978 resourceList := module.MaybeOutput("aapt2/res.list")
979 if resourceList.Rule != nil {
980 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000981 }
Colin Crossbec85302019-02-13 13:15:46 -0800982 overlayList := module.MaybeOutput("aapt2/overlay.list")
983 if overlayList.Rule != nil {
984 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
985 }
986
Anton Hansson53c88442019-03-18 15:53:16 +0000987 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
988 var prefix string
989 if d.overlayType == device {
990 prefix = "device:"
991 } else if d.overlayType == product {
992 prefix = "product:"
993 } else {
994 t.Fatalf("Unexpected overlayType %d", d.overlayType)
995 }
996 rroDirs = append(rroDirs, prefix+d.path.String())
997 }
Colin Crossbec85302019-02-13 13:15:46 -0800998
999 return resourceFiles, overlayFiles, rroDirs
1000 }
1001
1002 modules := []string{"foo", "bar", "lib", "lib2"}
1003 for _, module := range modules {
1004 resourceFiles, overlayFiles, rroDirs := getResources(module)
1005
1006 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
1007 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
1008 module, testCase.resourceFiles[module], resourceFiles)
1009 }
1010 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
1011 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
1012 module, testCase.overlayFiles[module], overlayFiles)
1013 }
1014 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +00001015 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -08001016 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +00001017 }
Colin Cross890ff552017-11-30 20:13:19 -08001018 }
Colin Cross890ff552017-11-30 20:13:19 -08001019 })
1020 }
1021}
Colin Crossd09b0b62018-04-18 11:06:47 -07001022
1023func TestAppSdkVersion(t *testing.T) {
1024 testCases := []struct {
1025 name string
1026 sdkVersion string
1027 platformSdkInt int
1028 platformSdkCodename string
1029 platformSdkFinal bool
1030 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +09001031 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -07001032 }{
1033 {
1034 name: "current final SDK",
1035 sdkVersion: "current",
1036 platformSdkInt: 27,
1037 platformSdkCodename: "REL",
1038 platformSdkFinal: true,
1039 expectedMinSdkVersion: "27",
1040 },
1041 {
1042 name: "current non-final SDK",
1043 sdkVersion: "current",
1044 platformSdkInt: 27,
1045 platformSdkCodename: "OMR1",
1046 platformSdkFinal: false,
1047 expectedMinSdkVersion: "OMR1",
1048 },
1049 {
1050 name: "default final SDK",
1051 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001052 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001053 platformSdkInt: 27,
1054 platformSdkCodename: "REL",
1055 platformSdkFinal: true,
1056 expectedMinSdkVersion: "27",
1057 },
1058 {
1059 name: "default non-final SDK",
1060 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001061 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001062 platformSdkInt: 27,
1063 platformSdkCodename: "OMR1",
1064 platformSdkFinal: false,
1065 expectedMinSdkVersion: "OMR1",
1066 },
1067 {
1068 name: "14",
1069 sdkVersion: "14",
1070 expectedMinSdkVersion: "14",
1071 },
1072 }
1073
1074 for _, moduleType := range []string{"android_app", "android_library"} {
1075 for _, test := range testCases {
1076 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001077 platformApiProp := ""
1078 if test.platformApis {
1079 platformApiProp = "platform_apis: true,"
1080 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001081 bp := fmt.Sprintf(`%s {
1082 name: "foo",
1083 srcs: ["a.java"],
1084 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001085 %s
1086 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001087
Colin Cross98be1bb2019-12-13 20:41:13 -08001088 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001089 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1090 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1091 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1092
Colin Cross98be1bb2019-12-13 20:41:13 -08001093 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -07001094
1095 run(t, ctx, config)
1096
1097 foo := ctx.ModuleForTests("foo", "android_common")
1098 link := foo.Output("package-res.apk")
1099 linkFlags := strings.Split(link.Args["flags"], " ")
1100 min := android.IndexList("--min-sdk-version", linkFlags)
1101 target := android.IndexList("--target-sdk-version", linkFlags)
1102
1103 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1104 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1105 }
1106
1107 gotMinSdkVersion := linkFlags[min+1]
1108 gotTargetSdkVersion := linkFlags[target+1]
1109
1110 if gotMinSdkVersion != test.expectedMinSdkVersion {
1111 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1112 test.expectedMinSdkVersion, gotMinSdkVersion)
1113 }
1114
1115 if gotTargetSdkVersion != test.expectedMinSdkVersion {
1116 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1117 test.expectedMinSdkVersion, gotTargetSdkVersion)
1118 }
1119 })
1120 }
1121 }
1122}
Colin Crossa4f08812018-10-02 22:03:40 -07001123
Paul Duffin50c217c2019-06-12 13:25:22 +01001124func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001125 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001126 cc_library {
1127 name: "libjni",
1128 system_shared_libs: [],
Colin Cross01fd7cc2020-02-19 16:54:04 -08001129 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001130 stl: "none",
1131 }
1132
1133 android_test {
1134 name: "test",
1135 sdk_version: "core_platform",
1136 jni_libs: ["libjni"],
1137 }
1138
1139 android_test {
1140 name: "test_first",
1141 sdk_version: "core_platform",
1142 compile_multilib: "first",
1143 jni_libs: ["libjni"],
1144 }
1145
1146 android_test {
1147 name: "test_both",
1148 sdk_version: "core_platform",
1149 compile_multilib: "both",
1150 jni_libs: ["libjni"],
1151 }
1152
1153 android_test {
1154 name: "test_32",
1155 sdk_version: "core_platform",
1156 compile_multilib: "32",
1157 jni_libs: ["libjni"],
1158 }
1159
1160 android_test {
1161 name: "test_64",
1162 sdk_version: "core_platform",
1163 compile_multilib: "64",
1164 jni_libs: ["libjni"],
1165 }
1166 `)
1167
1168 testCases := []struct {
1169 name string
1170 abis []string
1171 }{
1172 {"test", []string{"arm64-v8a"}},
1173 {"test_first", []string{"arm64-v8a"}},
1174 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1175 {"test_32", []string{"armeabi-v7a"}},
1176 {"test_64", []string{"arm64-v8a"}},
1177 }
1178
1179 for _, test := range testCases {
1180 t.Run(test.name, func(t *testing.T) {
1181 app := ctx.ModuleForTests(test.name, "android_common")
1182 jniLibZip := app.Output("jnilibs.zip")
1183 var abis []string
1184 args := strings.Fields(jniLibZip.Args["jarArgs"])
1185 for i := 0; i < len(args); i++ {
1186 if args[i] == "-P" {
1187 abis = append(abis, filepath.Base(args[i+1]))
1188 i++
1189 }
1190 }
1191 if !reflect.DeepEqual(abis, test.abis) {
1192 t.Errorf("want abis %v, got %v", test.abis, abis)
1193 }
1194 })
1195 }
1196}
1197
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001198func TestAppSdkVersionByPartition(t *testing.T) {
1199 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1200 android_app {
1201 name: "foo",
1202 srcs: ["a.java"],
1203 vendor: true,
1204 platform_apis: true,
1205 }
1206 `)
1207
1208 testJava(t, `
1209 android_app {
1210 name: "bar",
1211 srcs: ["b.java"],
1212 platform_apis: true,
1213 }
1214 `)
1215
1216 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001217 bp := `
1218 android_app {
1219 name: "foo",
1220 srcs: ["a.java"],
1221 product_specific: true,
1222 platform_apis: true,
1223 }
1224 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001225
1226 config := testAppConfig(nil, bp, nil)
1227 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001228 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001229 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 +09001230 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001231 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001232 }
1233 }
1234}
1235
Paul Duffin50c217c2019-06-12 13:25:22 +01001236func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001237 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001238 cc_library {
1239 name: "libjni",
1240 system_shared_libs: [],
1241 stl: "none",
Colin Cross1c93c292020-02-15 10:38:00 -08001242 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001243 }
1244
1245 android_app {
1246 name: "app",
1247 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001248 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001249 }
1250
1251 android_app {
1252 name: "app_noembed",
1253 jni_libs: ["libjni"],
1254 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001255 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001256 }
1257
1258 android_app {
1259 name: "app_embed",
1260 jni_libs: ["libjni"],
1261 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001262 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001263 }
1264
1265 android_test {
1266 name: "test",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001267 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001268 jni_libs: ["libjni"],
1269 }
1270
1271 android_test {
1272 name: "test_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001273 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001274 jni_libs: ["libjni"],
1275 use_embedded_native_libs: false,
1276 }
1277
1278 android_test_helper_app {
1279 name: "test_helper",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001280 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001281 jni_libs: ["libjni"],
1282 }
1283
1284 android_test_helper_app {
1285 name: "test_helper_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001286 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001287 jni_libs: ["libjni"],
1288 use_embedded_native_libs: false,
1289 }
1290 `)
1291
1292 testCases := []struct {
1293 name string
1294 packaged bool
1295 compressed bool
1296 }{
1297 {"app", false, false},
1298 {"app_noembed", false, false},
1299 {"app_embed", true, false},
1300 {"test", true, false},
1301 {"test_noembed", true, true},
1302 {"test_helper", true, false},
1303 {"test_helper_noembed", true, true},
1304 }
1305
1306 for _, test := range testCases {
1307 t.Run(test.name, func(t *testing.T) {
1308 app := ctx.ModuleForTests(test.name, "android_common")
1309 jniLibZip := app.MaybeOutput("jnilibs.zip")
1310 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1311 t.Errorf("expected jni packaged %v, got %v", w, g)
1312 }
1313
1314 if jniLibZip.Rule != nil {
1315 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1316 t.Errorf("expected jni compressed %v, got %v", w, g)
1317 }
Colin Cross01fd7cc2020-02-19 16:54:04 -08001318
1319 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1320 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1321 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001322 }
1323 })
1324 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001325}
1326
Colin Cross1dd9c442020-05-08 11:20:24 -07001327func TestJNISDK(t *testing.T) {
1328 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1329 cc_library {
1330 name: "libjni",
1331 system_shared_libs: [],
1332 stl: "none",
1333 sdk_version: "current",
1334 }
1335
1336 android_test {
1337 name: "app_platform",
1338 jni_libs: ["libjni"],
1339 platform_apis: true,
1340 }
1341
1342 android_test {
1343 name: "app_sdk",
1344 jni_libs: ["libjni"],
1345 sdk_version: "current",
1346 }
1347
1348 android_test {
1349 name: "app_force_platform",
1350 jni_libs: ["libjni"],
1351 sdk_version: "current",
1352 jni_uses_platform_apis: true,
1353 }
1354
1355 android_test {
1356 name: "app_force_sdk",
1357 jni_libs: ["libjni"],
1358 platform_apis: true,
1359 jni_uses_sdk_apis: true,
1360 }
Colin Crosseb032962020-05-13 11:05:02 -07001361
1362 cc_library {
1363 name: "libvendorjni",
1364 system_shared_libs: [],
1365 stl: "none",
1366 vendor: true,
1367 }
1368
1369 android_test {
1370 name: "app_vendor",
1371 jni_libs: ["libvendorjni"],
1372 sdk_version: "current",
1373 vendor: true,
1374 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001375 `)
1376
1377 testCases := []struct {
Colin Crosseb032962020-05-13 11:05:02 -07001378 name string
1379 sdkJNI bool
1380 vendorJNI bool
Colin Cross1dd9c442020-05-08 11:20:24 -07001381 }{
Colin Crosseb032962020-05-13 11:05:02 -07001382 {name: "app_platform"},
1383 {name: "app_sdk", sdkJNI: true},
1384 {name: "app_force_platform"},
1385 {name: "app_force_sdk", sdkJNI: true},
1386 {name: "app_vendor", vendorJNI: true},
Colin Cross1dd9c442020-05-08 11:20:24 -07001387 }
1388
Colin Crosseb032962020-05-13 11:05:02 -07001389 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1390 Output("libjni.so").Output.String()
1391 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1392 Output("libjni.so").Output.String()
1393 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1394 Output("libvendorjni.so").Output.String()
1395
Colin Cross1dd9c442020-05-08 11:20:24 -07001396 for _, test := range testCases {
1397 t.Run(test.name, func(t *testing.T) {
1398 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross1dd9c442020-05-08 11:20:24 -07001399
1400 jniLibZip := app.MaybeOutput("jnilibs.zip")
1401 if len(jniLibZip.Implicits) != 1 {
1402 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1403 }
1404 gotJNI := jniLibZip.Implicits[0].String()
1405
1406 if test.sdkJNI {
1407 if gotJNI != sdkJNI {
1408 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1409 }
Colin Crosseb032962020-05-13 11:05:02 -07001410 } else if test.vendorJNI {
1411 if gotJNI != vendorJNI {
1412 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1413 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001414 } else {
1415 if gotJNI != platformJNI {
1416 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1417 }
1418 }
1419 })
1420 }
1421
1422 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1423 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1424 android_test {
1425 name: "app_platform",
1426 platform_apis: true,
1427 jni_uses_platform_apis: true,
1428 }
1429 `)
1430 })
1431
1432 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1433 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1434 android_test {
1435 name: "app_sdk",
1436 sdk_version: "current",
1437 jni_uses_sdk_apis: true,
1438 }
1439 `)
1440 })
1441
1442}
1443
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001444func TestCertificates(t *testing.T) {
1445 testCases := []struct {
1446 name string
1447 bp string
1448 certificateOverride string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001449 expectedLineage string
1450 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001451 }{
1452 {
1453 name: "default",
1454 bp: `
1455 android_app {
1456 name: "foo",
1457 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001458 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001459 }
1460 `,
1461 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001462 expectedLineage: "",
1463 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001464 },
1465 {
1466 name: "module certificate property",
1467 bp: `
1468 android_app {
1469 name: "foo",
1470 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001471 certificate: ":new_certificate",
1472 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001473 }
1474
1475 android_app_certificate {
1476 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001477 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001478 }
1479 `,
1480 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001481 expectedLineage: "",
1482 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001483 },
1484 {
1485 name: "path certificate property",
1486 bp: `
1487 android_app {
1488 name: "foo",
1489 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001490 certificate: "expiredkey",
1491 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001492 }
1493 `,
1494 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001495 expectedLineage: "",
1496 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001497 },
1498 {
1499 name: "certificate overrides",
1500 bp: `
1501 android_app {
1502 name: "foo",
1503 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001504 certificate: "expiredkey",
1505 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001506 }
1507
1508 android_app_certificate {
1509 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001510 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001511 }
1512 `,
1513 certificateOverride: "foo:new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001514 expectedLineage: "",
1515 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1516 },
1517 {
1518 name: "certificate lineage",
1519 bp: `
1520 android_app {
1521 name: "foo",
1522 srcs: ["a.java"],
1523 certificate: ":new_certificate",
1524 lineage: "lineage.bin",
1525 sdk_version: "current",
1526 }
1527
1528 android_app_certificate {
1529 name: "new_certificate",
1530 certificate: "cert/new_cert",
1531 }
1532 `,
1533 certificateOverride: "",
1534 expectedLineage: "--lineage lineage.bin",
1535 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001536 },
1537 }
1538
1539 for _, test := range testCases {
1540 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001541 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001542 if test.certificateOverride != "" {
1543 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1544 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001545 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001546
1547 run(t, ctx, config)
1548 foo := ctx.ModuleForTests("foo", "android_common")
1549
1550 signapk := foo.Output("foo.apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001551 signCertificateFlags := signapk.Args["certificates"]
1552 if test.expectedCertificate != signCertificateFlags {
1553 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1554 }
1555
1556 signFlags := signapk.Args["flags"]
1557 if test.expectedLineage != signFlags {
1558 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001559 }
1560 })
1561 }
1562}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001563
Songchun Fan688de9a2020-03-24 20:32:24 -07001564func TestRequestV4SigningFlag(t *testing.T) {
1565 testCases := []struct {
1566 name string
1567 bp string
1568 expected string
1569 }{
1570 {
1571 name: "default",
1572 bp: `
1573 android_app {
1574 name: "foo",
1575 srcs: ["a.java"],
1576 sdk_version: "current",
1577 }
1578 `,
1579 expected: "",
1580 },
1581 {
1582 name: "default",
1583 bp: `
1584 android_app {
1585 name: "foo",
1586 srcs: ["a.java"],
1587 sdk_version: "current",
1588 v4_signature: false,
1589 }
1590 `,
1591 expected: "",
1592 },
1593 {
1594 name: "module certificate property",
1595 bp: `
1596 android_app {
1597 name: "foo",
1598 srcs: ["a.java"],
1599 sdk_version: "current",
1600 v4_signature: true,
1601 }
1602 `,
1603 expected: "--enable-v4",
1604 },
1605 }
1606
1607 for _, test := range testCases {
1608 t.Run(test.name, func(t *testing.T) {
1609 config := testAppConfig(nil, test.bp, nil)
1610 ctx := testContext()
1611
1612 run(t, ctx, config)
1613 foo := ctx.ModuleForTests("foo", "android_common")
1614
1615 signapk := foo.Output("foo.apk")
1616 signFlags := signapk.Args["flags"]
1617 if test.expected != signFlags {
1618 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1619 }
1620 })
1621 }
1622}
1623
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001624func TestPackageNameOverride(t *testing.T) {
1625 testCases := []struct {
1626 name string
1627 bp string
1628 packageNameOverride string
1629 expected []string
1630 }{
1631 {
1632 name: "default",
1633 bp: `
1634 android_app {
1635 name: "foo",
1636 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001637 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001638 }
1639 `,
1640 packageNameOverride: "",
1641 expected: []string{
1642 buildDir + "/.intermediates/foo/android_common/foo.apk",
1643 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1644 },
1645 },
1646 {
1647 name: "overridden",
1648 bp: `
1649 android_app {
1650 name: "foo",
1651 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001652 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001653 }
1654 `,
1655 packageNameOverride: "foo:bar",
1656 expected: []string{
1657 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001658 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001659 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1660 },
1661 },
1662 }
1663
1664 for _, test := range testCases {
1665 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001666 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001667 if test.packageNameOverride != "" {
1668 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1669 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001670 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001671
1672 run(t, ctx, config)
1673 foo := ctx.ModuleForTests("foo", "android_common")
1674
1675 outputs := foo.AllOutputs()
1676 outputMap := make(map[string]bool)
1677 for _, o := range outputs {
1678 outputMap[o] = true
1679 }
1680 for _, e := range test.expected {
1681 if _, exist := outputMap[e]; !exist {
1682 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1683 }
1684 }
1685 })
1686 }
1687}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001688
1689func TestInstrumentationTargetOverridden(t *testing.T) {
1690 bp := `
1691 android_app {
1692 name: "foo",
1693 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001694 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001695 }
1696
1697 android_test {
1698 name: "bar",
1699 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001700 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001701 }
1702 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001703 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001704 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001705 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001706
1707 run(t, ctx, config)
1708
1709 bar := ctx.ModuleForTests("bar", "android_common")
1710 res := bar.Output("package-res.apk")
1711 aapt2Flags := res.Args["flags"]
1712 e := "--rename-instrumentation-target-package org.dandroid.bp"
1713 if !strings.Contains(aapt2Flags, e) {
1714 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1715 }
1716}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001717
1718func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001719 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001720 android_app {
1721 name: "foo",
1722 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001723 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001724 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001725 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001726 }
1727
1728 override_android_app {
1729 name: "bar",
1730 base: "foo",
1731 certificate: ":new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001732 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001733 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001734 }
1735
1736 android_app_certificate {
1737 name: "new_certificate",
1738 certificate: "cert/new_cert",
1739 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001740
1741 override_android_app {
1742 name: "baz",
1743 base: "foo",
1744 package_name: "org.dandroid.bp",
1745 }
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00001746
1747 override_android_app {
1748 name: "baz_no_rename_resources",
1749 base: "foo",
1750 package_name: "org.dandroid.bp",
1751 rename_resources_package: false,
1752 }
1753
1754 android_app {
1755 name: "foo_no_rename_resources",
1756 srcs: ["a.java"],
1757 certificate: "expiredkey",
1758 overrides: ["qux"],
1759 rename_resources_package: false,
1760 sdk_version: "current",
1761 }
1762
1763 override_android_app {
1764 name: "baz_base_no_rename_resources",
1765 base: "foo_no_rename_resources",
1766 package_name: "org.dandroid.bp",
1767 }
1768
1769 override_android_app {
1770 name: "baz_override_base_rename_resources",
1771 base: "foo_no_rename_resources",
1772 package_name: "org.dandroid.bp",
1773 rename_resources_package: true,
1774 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001775 `)
1776
1777 expectedVariants := []struct {
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00001778 name string
1779 moduleName string
1780 variantName string
1781 apkName string
1782 apkPath string
1783 certFlag string
1784 lineageFlag string
1785 overrides []string
1786 packageFlag string
1787 renameResources bool
1788 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001789 }{
1790 {
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00001791 name: "foo",
1792 moduleName: "foo",
1793 variantName: "android_common",
1794 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
1795 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1796 lineageFlag: "",
1797 overrides: []string{"qux"},
1798 packageFlag: "",
1799 renameResources: false,
1800 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001801 },
1802 {
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00001803 name: "foo",
1804 moduleName: "bar",
1805 variantName: "android_common_bar",
1806 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
1807 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1808 lineageFlag: "--lineage lineage.bin",
1809 overrides: []string{"qux", "foo"},
1810 packageFlag: "",
1811 renameResources: false,
1812 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001813 },
1814 {
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00001815 name: "foo",
1816 moduleName: "baz",
1817 variantName: "android_common_baz",
1818 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
1819 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1820 lineageFlag: "",
1821 overrides: []string{"qux", "foo"},
1822 packageFlag: "org.dandroid.bp",
1823 renameResources: true,
1824 logging_parent: "",
1825 },
1826 {
1827 name: "foo",
1828 moduleName: "baz_no_rename_resources",
1829 variantName: "android_common_baz_no_rename_resources",
1830 apkPath: "/target/product/test_device/system/app/baz_no_rename_resources/baz_no_rename_resources.apk",
1831 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1832 lineageFlag: "",
1833 overrides: []string{"qux", "foo"},
1834 packageFlag: "org.dandroid.bp",
1835 renameResources: false,
1836 logging_parent: "",
1837 },
1838 {
1839 name: "foo_no_rename_resources",
1840 moduleName: "baz_base_no_rename_resources",
1841 variantName: "android_common_baz_base_no_rename_resources",
1842 apkPath: "/target/product/test_device/system/app/baz_base_no_rename_resources/baz_base_no_rename_resources.apk",
1843 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1844 lineageFlag: "",
1845 overrides: []string{"qux", "foo_no_rename_resources"},
1846 packageFlag: "org.dandroid.bp",
1847 renameResources: false,
1848 logging_parent: "",
1849 },
1850 {
1851 name: "foo_no_rename_resources",
1852 moduleName: "baz_override_base_rename_resources",
1853 variantName: "android_common_baz_override_base_rename_resources",
1854 apkPath: "/target/product/test_device/system/app/baz_override_base_rename_resources/baz_override_base_rename_resources.apk",
1855 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1856 lineageFlag: "",
1857 overrides: []string{"qux", "foo_no_rename_resources"},
1858 packageFlag: "org.dandroid.bp",
1859 renameResources: true,
1860 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001861 },
1862 }
1863 for _, expected := range expectedVariants {
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00001864 variant := ctx.ModuleForTests(expected.name, expected.variantName)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001865
1866 // Check the final apk name
1867 outputs := variant.AllOutputs()
1868 expectedApkPath := buildDir + expected.apkPath
1869 found := false
1870 for _, o := range outputs {
1871 if o == expectedApkPath {
1872 found = true
1873 break
1874 }
1875 }
1876 if !found {
1877 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1878 }
1879
1880 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001881 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001882 certFlag := signapk.Args["certificates"]
1883 if expected.certFlag != certFlag {
1884 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1885 }
1886
1887 // Check the lineage flags
1888 lineageFlag := signapk.Args["flags"]
1889 if expected.lineageFlag != lineageFlag {
1890 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001891 }
1892
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001893 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001894 mod := variant.Module().(*AndroidApp)
1895 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1896 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1897 expected.overrides, mod.appProperties.Overrides)
1898 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001899
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001900 // Test Overridable property: Logging_parent
1901 logging_parent := mod.aapt.LoggingParent
1902 if expected.logging_parent != logging_parent {
1903 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1904 expected.logging_parent, logging_parent)
1905 }
1906
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001907 // Check the package renaming flag, if exists.
1908 res := variant.Output("package-res.apk")
1909 aapt2Flags := res.Args["flags"]
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00001910 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1911 expectedPackage := expected.packageFlag
1912 if !expected.renameResources {
1913 expectedPackage = ""
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001914 }
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00001915 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expectedPackage)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001916 }
1917}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001918
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001919func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001920 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001921 android_app {
1922 name: "foo",
1923 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001924 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001925 }
1926
1927 override_android_app {
1928 name: "bar",
1929 base: "foo",
1930 package_name: "org.dandroid.bp",
1931 }
1932
1933 android_test {
1934 name: "baz",
1935 srcs: ["b.java"],
1936 instrumentation_for: "foo",
1937 }
1938
1939 android_test {
1940 name: "qux",
1941 srcs: ["b.java"],
1942 instrumentation_for: "bar",
1943 }
1944 `)
1945
1946 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1947 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1948 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1949 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1950 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1951 }
1952
1953 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1954 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1955 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1956 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1957 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1958 }
1959}
1960
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001961func TestOverrideAndroidTest(t *testing.T) {
1962 ctx, _ := testJava(t, `
1963 android_app {
1964 name: "foo",
1965 srcs: ["a.java"],
1966 package_name: "com.android.foo",
1967 sdk_version: "current",
1968 }
1969
1970 override_android_app {
1971 name: "bar",
1972 base: "foo",
1973 package_name: "com.android.bar",
1974 }
1975
1976 android_test {
1977 name: "foo_test",
1978 srcs: ["b.java"],
1979 instrumentation_for: "foo",
1980 }
1981
1982 override_android_test {
1983 name: "bar_test",
1984 base: "foo_test",
1985 package_name: "com.android.bar.test",
1986 instrumentation_for: "bar",
1987 instrumentation_target_package: "com.android.bar",
1988 }
1989 `)
1990
1991 expectedVariants := []struct {
1992 moduleName string
1993 variantName string
1994 apkPath string
1995 overrides []string
1996 targetVariant string
1997 packageFlag string
1998 targetPackageFlag string
1999 }{
2000 {
2001 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002002 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002003 overrides: nil,
2004 targetVariant: "android_common",
2005 packageFlag: "",
2006 targetPackageFlag: "",
2007 },
2008 {
2009 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002010 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002011 overrides: []string{"foo_test"},
2012 targetVariant: "android_common_bar",
2013 packageFlag: "com.android.bar.test",
2014 targetPackageFlag: "com.android.bar",
2015 },
2016 }
2017 for _, expected := range expectedVariants {
2018 variant := ctx.ModuleForTests("foo_test", expected.variantName)
2019
2020 // Check the final apk name
2021 outputs := variant.AllOutputs()
2022 expectedApkPath := buildDir + expected.apkPath
2023 found := false
2024 for _, o := range outputs {
2025 if o == expectedApkPath {
2026 found = true
2027 break
2028 }
2029 }
2030 if !found {
2031 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
2032 }
2033
2034 // Check if the overrides field values are correctly aggregated.
2035 mod := variant.Module().(*AndroidTest)
2036 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
2037 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
2038 expected.overrides, mod.appProperties.Overrides)
2039 }
2040
2041 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
2042 javac := variant.Rule("javac")
2043 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
2044 if !strings.Contains(javac.Args["classpath"], turbine) {
2045 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
2046 }
2047
2048 // Check aapt2 flags.
2049 res := variant.Output("package-res.apk")
2050 aapt2Flags := res.Args["flags"]
2051 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00002052 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expected.packageFlag)
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002053 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
2054 }
2055}
2056
Jaewoong Jung39982342020-01-14 10:27:18 -08002057func TestAndroidTest_FixTestConfig(t *testing.T) {
2058 ctx, _ := testJava(t, `
2059 android_app {
2060 name: "foo",
2061 srcs: ["a.java"],
2062 package_name: "com.android.foo",
2063 sdk_version: "current",
2064 }
2065
2066 android_test {
2067 name: "foo_test",
2068 srcs: ["b.java"],
2069 instrumentation_for: "foo",
2070 }
2071
2072 android_test {
2073 name: "bar_test",
2074 srcs: ["b.java"],
2075 package_name: "com.android.bar.test",
2076 instrumentation_for: "foo",
2077 }
2078
2079 override_android_test {
2080 name: "baz_test",
2081 base: "foo_test",
2082 package_name: "com.android.baz.test",
2083 }
2084 `)
2085
2086 testCases := []struct {
2087 moduleName string
2088 variantName string
2089 expectedFlags []string
2090 }{
2091 {
2092 moduleName: "foo_test",
2093 variantName: "android_common",
2094 },
2095 {
2096 moduleName: "bar_test",
2097 variantName: "android_common",
2098 expectedFlags: []string{
2099 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
2100 "--package-name com.android.bar.test",
2101 },
2102 },
2103 {
2104 moduleName: "foo_test",
2105 variantName: "android_common_baz_test",
2106 expectedFlags: []string{
2107 "--manifest " + buildDir +
2108 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
2109 "--package-name com.android.baz.test",
2110 "--test-file-name baz_test.apk",
2111 },
2112 },
2113 }
2114
2115 for _, test := range testCases {
2116 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
2117 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
2118
2119 if len(test.expectedFlags) > 0 {
2120 if params.Rule == nil {
2121 t.Errorf("test_config_fixer was expected to run, but didn't")
2122 } else {
2123 for _, flag := range test.expectedFlags {
2124 if !strings.Contains(params.RuleParams.Command, flag) {
2125 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2126 }
2127 }
2128 }
2129 } else {
2130 if params.Rule != nil {
2131 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2132 }
2133 }
2134
2135 }
2136}
2137
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002138func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002139 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002140 android_app_import {
2141 name: "foo",
2142 apk: "prebuilts/apk/app.apk",
2143 certificate: "platform",
2144 dex_preopt: {
2145 enabled: true,
2146 },
2147 }
2148 `)
2149
2150 variant := ctx.ModuleForTests("foo", "android_common")
2151
2152 // Check dexpreopt outputs.
2153 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2154 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2155 t.Errorf("can't find dexpreopt outputs")
2156 }
2157
2158 // Check cert signing flag.
2159 signedApk := variant.Output("signed/foo.apk")
2160 signingFlag := signedApk.Args["certificates"]
2161 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2162 if expected != signingFlag {
2163 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2164 }
2165}
2166
2167func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002168 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002169 android_app_import {
2170 name: "foo",
2171 apk: "prebuilts/apk/app.apk",
2172 certificate: "platform",
2173 dex_preopt: {
2174 enabled: false,
2175 },
2176 }
2177 `)
2178
2179 variant := ctx.ModuleForTests("foo", "android_common")
2180
2181 // Check dexpreopt outputs. They shouldn't exist.
2182 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2183 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2184 t.Errorf("dexpreopt shouldn't have run.")
2185 }
2186}
2187
2188func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002189 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002190 android_app_import {
2191 name: "foo",
2192 apk: "prebuilts/apk/app.apk",
2193 presigned: true,
2194 dex_preopt: {
2195 enabled: true,
2196 },
2197 }
2198 `)
2199
2200 variant := ctx.ModuleForTests("foo", "android_common")
2201
2202 // Check dexpreopt outputs.
2203 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2204 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2205 t.Errorf("can't find dexpreopt outputs")
2206 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002207 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002208 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2209 t.Errorf("signing rule shouldn't be included.")
2210 }
2211 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2212 t.Errorf("can't find aligning rule")
2213 }
2214}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002215
Liz Kammer2bc57f62020-05-13 15:49:21 -07002216func TestAndroidAppImport_SigningLineage(t *testing.T) {
2217 ctx, _ := testJava(t, `
2218 android_app_import {
2219 name: "foo",
2220 apk: "prebuilts/apk/app.apk",
2221 certificate: "platform",
2222 lineage: "lineage.bin",
2223 }
2224 `)
2225
2226 variant := ctx.ModuleForTests("foo", "android_common")
2227
2228 // Check cert signing lineage flag.
2229 signedApk := variant.Output("signed/foo.apk")
2230 signingFlag := signedApk.Args["flags"]
2231 expected := "--lineage lineage.bin"
2232 if expected != signingFlag {
2233 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2234 }
2235}
2236
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002237func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2238 ctx, _ := testJava(t, `
2239 android_app_import {
2240 name: "foo",
2241 apk: "prebuilts/apk/app.apk",
2242 default_dev_cert: true,
2243 dex_preopt: {
2244 enabled: true,
2245 },
2246 }
2247 `)
2248
2249 variant := ctx.ModuleForTests("foo", "android_common")
2250
2251 // Check dexpreopt outputs.
2252 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2253 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2254 t.Errorf("can't find dexpreopt outputs")
2255 }
2256
2257 // Check cert signing flag.
2258 signedApk := variant.Output("signed/foo.apk")
2259 signingFlag := signedApk.Args["certificates"]
2260 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2261 if expected != signingFlag {
2262 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2263 }
2264}
2265
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002266func TestAndroidAppImport_DpiVariants(t *testing.T) {
2267 bp := `
2268 android_app_import {
2269 name: "foo",
2270 apk: "prebuilts/apk/app.apk",
2271 dpi_variants: {
2272 xhdpi: {
2273 apk: "prebuilts/apk/app_xhdpi.apk",
2274 },
2275 xxhdpi: {
2276 apk: "prebuilts/apk/app_xxhdpi.apk",
2277 },
2278 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002279 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002280 dex_preopt: {
2281 enabled: true,
2282 },
2283 }
2284 `
2285 testCases := []struct {
2286 name string
2287 aaptPreferredConfig *string
2288 aaptPrebuiltDPI []string
2289 expected string
2290 }{
2291 {
2292 name: "no preferred",
2293 aaptPreferredConfig: nil,
2294 aaptPrebuiltDPI: []string{},
2295 expected: "prebuilts/apk/app.apk",
2296 },
2297 {
2298 name: "AAPTPreferredConfig matches",
2299 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002300 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002301 expected: "prebuilts/apk/app_xhdpi.apk",
2302 },
2303 {
2304 name: "AAPTPrebuiltDPI matches",
2305 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2306 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2307 expected: "prebuilts/apk/app_xxhdpi.apk",
2308 },
2309 {
2310 name: "non-first AAPTPrebuiltDPI matches",
2311 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2312 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2313 expected: "prebuilts/apk/app_xhdpi.apk",
2314 },
2315 {
2316 name: "no matches",
2317 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2318 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2319 expected: "prebuilts/apk/app.apk",
2320 },
2321 }
2322
2323 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2324 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002325 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002326 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2327 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002328 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002329
2330 run(t, ctx, config)
2331
2332 variant := ctx.ModuleForTests("foo", "android_common")
2333 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2334 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2335 if len(matches) != 2 {
2336 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2337 }
2338 if test.expected != matches[1] {
2339 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2340 }
2341 }
2342}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002343
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002344func TestAndroidAppImport_Filename(t *testing.T) {
2345 ctx, config := testJava(t, `
2346 android_app_import {
2347 name: "foo",
2348 apk: "prebuilts/apk/app.apk",
2349 presigned: true,
2350 }
2351
2352 android_app_import {
2353 name: "bar",
2354 apk: "prebuilts/apk/app.apk",
2355 presigned: true,
2356 filename: "bar_sample.apk"
2357 }
2358 `)
2359
2360 testCases := []struct {
2361 name string
2362 expected string
2363 }{
2364 {
2365 name: "foo",
2366 expected: "foo.apk",
2367 },
2368 {
2369 name: "bar",
2370 expected: "bar_sample.apk",
2371 },
2372 }
2373
2374 for _, test := range testCases {
2375 variant := ctx.ModuleForTests(test.name, "android_common")
2376 if variant.MaybeOutput(test.expected).Rule == nil {
2377 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2378 }
2379
2380 a := variant.Module().(*AndroidAppImport)
2381 expectedValues := []string{test.expected}
2382 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002383 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002384 if !reflect.DeepEqual(actualValues, expectedValues) {
2385 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2386 actualValues, expectedValues)
2387 }
2388 }
2389}
2390
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002391func TestAndroidAppImport_ArchVariants(t *testing.T) {
2392 // The test config's target arch is ARM64.
2393 testCases := []struct {
2394 name string
2395 bp string
2396 expected string
2397 }{
2398 {
2399 name: "matching arch",
2400 bp: `
2401 android_app_import {
2402 name: "foo",
2403 apk: "prebuilts/apk/app.apk",
2404 arch: {
2405 arm64: {
2406 apk: "prebuilts/apk/app_arm64.apk",
2407 },
2408 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002409 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002410 dex_preopt: {
2411 enabled: true,
2412 },
2413 }
2414 `,
2415 expected: "prebuilts/apk/app_arm64.apk",
2416 },
2417 {
2418 name: "no matching arch",
2419 bp: `
2420 android_app_import {
2421 name: "foo",
2422 apk: "prebuilts/apk/app.apk",
2423 arch: {
2424 arm: {
2425 apk: "prebuilts/apk/app_arm.apk",
2426 },
2427 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002428 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002429 dex_preopt: {
2430 enabled: true,
2431 },
2432 }
2433 `,
2434 expected: "prebuilts/apk/app.apk",
2435 },
2436 }
2437
2438 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2439 for _, test := range testCases {
2440 ctx, _ := testJava(t, test.bp)
2441
2442 variant := ctx.ModuleForTests("foo", "android_common")
2443 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2444 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2445 if len(matches) != 2 {
2446 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2447 }
2448 if test.expected != matches[1] {
2449 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2450 }
2451 }
2452}
2453
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002454func TestAndroidTestImport(t *testing.T) {
2455 ctx, config := testJava(t, `
2456 android_test_import {
2457 name: "foo",
2458 apk: "prebuilts/apk/app.apk",
2459 presigned: true,
2460 data: [
2461 "testdata/data",
2462 ],
2463 }
2464 `)
2465
2466 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2467
2468 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002469 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002470 expected := []string{"tests"}
2471 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2472 if !reflect.DeepEqual(expected, actual) {
2473 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2474 }
2475 expected = []string{"testdata/data:testdata/data"}
2476 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2477 if !reflect.DeepEqual(expected, actual) {
2478 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2479 }
2480}
2481
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002482func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2483 ctx, _ := testJava(t, `
2484 android_test_import {
2485 name: "foo",
2486 apk: "prebuilts/apk/app.apk",
2487 certificate: "cert/new_cert",
2488 data: [
2489 "testdata/data",
2490 ],
2491 }
2492
2493 android_test_import {
2494 name: "foo_presigned",
2495 apk: "prebuilts/apk/app.apk",
2496 presigned: true,
2497 data: [
2498 "testdata/data",
2499 ],
2500 }
2501 `)
2502
2503 variant := ctx.ModuleForTests("foo", "android_common")
2504 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2505 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2506 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2507 }
2508
2509 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2510 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2511 if jniRule != android.Cp.String() {
2512 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2513 }
Liz Kammer7e20dda2020-05-20 14:36:30 -07002514 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2515 t.Errorf("Presigned test apk should be aligned")
2516 }
2517}
2518
2519func TestAndroidTestImport_Preprocessed(t *testing.T) {
2520 ctx, _ := testJava(t, `
2521 android_test_import {
2522 name: "foo",
2523 apk: "prebuilts/apk/app.apk",
2524 presigned: true,
2525 preprocessed: true,
2526 }
2527
2528 android_test_import {
2529 name: "foo_cert",
2530 apk: "prebuilts/apk/app.apk",
2531 certificate: "cert/new_cert",
2532 preprocessed: true,
2533 }
2534 `)
2535
2536 testModules := []string{"foo", "foo_cert"}
2537 for _, m := range testModules {
2538 apkName := m + ".apk"
2539 variant := ctx.ModuleForTests(m, "android_common")
2540 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2541 if jniRule != android.Cp.String() {
2542 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2543 }
2544
2545 // Make sure signing and aligning were skipped.
2546 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2547 t.Errorf("signing rule shouldn't be included for preprocessed.")
2548 }
2549 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2550 t.Errorf("aligning rule shouldn't be for preprocessed")
2551 }
2552 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002553}
2554
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002555func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002556 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002557 cc_library {
2558 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002559 sdk_version: "current",
2560 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002561 }
2562
2563 android_test {
2564 name: "stl",
2565 jni_libs: ["libjni"],
2566 compile_multilib: "both",
2567 sdk_version: "current",
2568 stl: "c++_shared",
2569 }
2570
2571 android_test {
2572 name: "system",
2573 jni_libs: ["libjni"],
2574 compile_multilib: "both",
2575 sdk_version: "current",
2576 }
2577 `)
2578
2579 testCases := []struct {
2580 name string
2581 jnis []string
2582 }{
2583 {"stl",
2584 []string{
2585 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002586 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002587 },
2588 },
2589 {"system",
2590 []string{
2591 "libjni.so",
2592 },
2593 },
2594 }
2595
2596 for _, test := range testCases {
2597 t.Run(test.name, func(t *testing.T) {
2598 app := ctx.ModuleForTests(test.name, "android_common")
2599 jniLibZip := app.Output("jnilibs.zip")
2600 var jnis []string
2601 args := strings.Fields(jniLibZip.Args["jarArgs"])
2602 for i := 0; i < len(args); i++ {
2603 if args[i] == "-f" {
2604 jnis = append(jnis, args[i+1])
2605 i += 1
2606 }
2607 }
2608 jnisJoined := strings.Join(jnis, " ")
2609 for _, jni := range test.jnis {
2610 if !strings.Contains(jnisJoined, jni) {
2611 t.Errorf("missing jni %q in %q", jni, jnis)
2612 }
2613 }
2614 })
2615 }
2616}
Colin Cross50ddcc42019-05-16 12:28:22 -07002617
2618func TestUsesLibraries(t *testing.T) {
2619 bp := `
2620 java_sdk_library {
2621 name: "foo",
2622 srcs: ["a.java"],
2623 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002624 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002625 }
2626
2627 java_sdk_library {
Paul Duffin64e61992020-05-15 10:20:31 +01002628 name: "qux",
2629 srcs: ["a.java"],
2630 api_packages: ["qux"],
2631 sdk_version: "current",
2632 }
2633
2634 java_sdk_library {
2635 name: "quuz",
2636 srcs: ["a.java"],
2637 api_packages: ["quuz"],
2638 sdk_version: "current",
2639 }
2640
2641 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002642 name: "bar",
2643 srcs: ["a.java"],
2644 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002645 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002646 }
2647
2648 android_app {
2649 name: "app",
2650 srcs: ["a.java"],
Paul Duffin64e61992020-05-15 10:20:31 +01002651 libs: ["qux", "quuz.stubs"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002652 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002653 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002654 optional_uses_libs: [
2655 "bar",
2656 "baz",
2657 ],
2658 }
2659
2660 android_app_import {
2661 name: "prebuilt",
2662 apk: "prebuilts/apk/app.apk",
2663 certificate: "platform",
2664 uses_libs: ["foo"],
2665 optional_uses_libs: [
2666 "bar",
2667 "baz",
2668 ],
2669 }
2670 `
2671
Colin Cross98be1bb2019-12-13 20:41:13 -08002672 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002673 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2674
Colin Cross98be1bb2019-12-13 20:41:13 -08002675 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002676
2677 run(t, ctx, config)
2678
2679 app := ctx.ModuleForTests("app", "android_common")
2680 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2681
Paul Duffin64e61992020-05-15 10:20:31 +01002682 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2683 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2684 if w := "--uses-library qux"; !strings.Contains(manifestFixerArgs, w) {
2685 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2686 }
2687 if w := "--uses-library quuz"; !strings.Contains(manifestFixerArgs, w) {
2688 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2689 }
2690
Colin Cross50ddcc42019-05-16 12:28:22 -07002691 // Test that all libraries are verified
2692 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2693 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2694 t.Errorf("wanted %q in %q", w, cmd)
2695 }
2696
2697 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2698 t.Errorf("wanted %q in %q", w, cmd)
2699 }
2700
2701 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2702
2703 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2704 t.Errorf("wanted %q in %q", w, cmd)
2705 }
2706
2707 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2708 t.Errorf("wanted %q in %q", w, cmd)
2709 }
2710
2711 // Test that only present libraries are preopted
2712 cmd = app.Rule("dexpreopt").RuleParams.Command
2713
2714 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2715 t.Errorf("wanted %q in %q", w, cmd)
2716 }
2717
2718 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2719
2720 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2721 t.Errorf("wanted %q in %q", w, cmd)
2722 }
2723}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002724
2725func TestCodelessApp(t *testing.T) {
2726 testCases := []struct {
2727 name string
2728 bp string
2729 noCode bool
2730 }{
2731 {
2732 name: "normal",
2733 bp: `
2734 android_app {
2735 name: "foo",
2736 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002737 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002738 }
2739 `,
2740 noCode: false,
2741 },
2742 {
2743 name: "app without sources",
2744 bp: `
2745 android_app {
2746 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002747 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002748 }
2749 `,
2750 noCode: true,
2751 },
2752 {
2753 name: "app with libraries",
2754 bp: `
2755 android_app {
2756 name: "foo",
2757 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002758 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002759 }
2760
2761 java_library {
2762 name: "lib",
2763 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002764 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002765 }
2766 `,
2767 noCode: false,
2768 },
2769 {
2770 name: "app with sourceless libraries",
2771 bp: `
2772 android_app {
2773 name: "foo",
2774 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002775 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002776 }
2777
2778 java_library {
2779 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002780 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002781 }
2782 `,
2783 // TODO(jungjw): this should probably be true
2784 noCode: false,
2785 },
2786 }
2787
2788 for _, test := range testCases {
2789 t.Run(test.name, func(t *testing.T) {
2790 ctx := testApp(t, test.bp)
2791
2792 foo := ctx.ModuleForTests("foo", "android_common")
2793 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2794 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2795 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2796 }
2797 })
2798 }
2799}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002800
2801func TestEmbedNotice(t *testing.T) {
Colin Crossf5f663b2020-06-07 16:58:18 -07002802 ctx, _ := testJavaWithFS(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002803 android_app {
2804 name: "foo",
2805 srcs: ["a.java"],
2806 static_libs: ["javalib"],
2807 jni_libs: ["libjni"],
2808 notice: "APP_NOTICE",
2809 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002810 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002811 }
2812
2813 // No embed_notice flag
2814 android_app {
2815 name: "bar",
2816 srcs: ["a.java"],
2817 jni_libs: ["libjni"],
2818 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002819 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002820 }
2821
2822 // No NOTICE files
2823 android_app {
2824 name: "baz",
2825 srcs: ["a.java"],
2826 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002827 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002828 }
2829
2830 cc_library {
2831 name: "libjni",
2832 system_shared_libs: [],
2833 stl: "none",
2834 notice: "LIB_NOTICE",
Colin Cross1c93c292020-02-15 10:38:00 -08002835 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002836 }
2837
2838 java_library {
2839 name: "javalib",
2840 srcs: [
2841 ":gen",
2842 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002843 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002844 }
2845
2846 genrule {
2847 name: "gen",
2848 tools: ["gentool"],
2849 out: ["gen.java"],
2850 notice: "GENRULE_NOTICE",
2851 }
2852
2853 java_binary_host {
2854 name: "gentool",
2855 srcs: ["b.java"],
2856 notice: "TOOL_NOTICE",
2857 }
Colin Crossf5f663b2020-06-07 16:58:18 -07002858 `, map[string][]byte{
2859 "APP_NOTICE": nil,
2860 "GENRULE_NOTICE": nil,
2861 "LIB_NOTICE": nil,
2862 "TOOL_NOTICE": nil,
2863 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002864
2865 // foo has NOTICE files to process, and embed_notices is true.
2866 foo := ctx.ModuleForTests("foo", "android_common")
2867 // verify merge notices rule.
2868 mergeNotices := foo.Rule("mergeNoticesRule")
2869 noticeInputs := mergeNotices.Inputs.Strings()
2870 // TOOL_NOTICE should be excluded as it's a host module.
2871 if len(mergeNotices.Inputs) != 3 {
2872 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2873 }
2874 if !inList("APP_NOTICE", noticeInputs) {
2875 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2876 }
2877 if !inList("LIB_NOTICE", noticeInputs) {
2878 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2879 }
2880 if !inList("GENRULE_NOTICE", noticeInputs) {
2881 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2882 }
2883 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2884 res := foo.Output("package-res.apk")
2885 aapt2Flags := res.Args["flags"]
2886 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2887 if !strings.Contains(aapt2Flags, e) {
2888 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2889 }
2890
2891 // bar has NOTICE files to process, but embed_notices is not set.
2892 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002893 res = bar.Output("package-res.apk")
2894 aapt2Flags = res.Args["flags"]
2895 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2896 if strings.Contains(aapt2Flags, e) {
2897 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002898 }
2899
2900 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2901 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002902 res = baz.Output("package-res.apk")
2903 aapt2Flags = res.Args["flags"]
2904 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2905 if strings.Contains(aapt2Flags, e) {
2906 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002907 }
2908}
Colin Cross53a87f52019-06-25 13:35:30 -07002909
2910func TestUncompressDex(t *testing.T) {
2911 testCases := []struct {
2912 name string
2913 bp string
2914
2915 uncompressedPlatform bool
2916 uncompressedUnbundled bool
2917 }{
2918 {
2919 name: "normal",
2920 bp: `
2921 android_app {
2922 name: "foo",
2923 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002924 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002925 }
2926 `,
2927 uncompressedPlatform: true,
2928 uncompressedUnbundled: false,
2929 },
2930 {
2931 name: "use_embedded_dex",
2932 bp: `
2933 android_app {
2934 name: "foo",
2935 use_embedded_dex: true,
2936 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002937 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002938 }
2939 `,
2940 uncompressedPlatform: true,
2941 uncompressedUnbundled: true,
2942 },
2943 {
2944 name: "privileged",
2945 bp: `
2946 android_app {
2947 name: "foo",
2948 privileged: true,
2949 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002950 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002951 }
2952 `,
2953 uncompressedPlatform: true,
2954 uncompressedUnbundled: true,
2955 },
David Srbecky98c71222020-05-20 22:20:28 +01002956 {
2957 name: "normal_uncompress_dex_true",
2958 bp: `
2959 android_app {
2960 name: "foo",
2961 srcs: ["a.java"],
2962 sdk_version: "current",
2963 uncompress_dex: true,
2964 }
2965 `,
2966 uncompressedPlatform: true,
2967 uncompressedUnbundled: true,
2968 },
2969 {
2970 name: "normal_uncompress_dex_false",
2971 bp: `
2972 android_app {
2973 name: "foo",
2974 srcs: ["a.java"],
2975 sdk_version: "current",
2976 uncompress_dex: false,
2977 }
2978 `,
2979 uncompressedPlatform: false,
2980 uncompressedUnbundled: false,
2981 },
Colin Cross53a87f52019-06-25 13:35:30 -07002982 }
2983
2984 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2985 t.Helper()
2986
Colin Cross98be1bb2019-12-13 20:41:13 -08002987 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002988 if unbundled {
2989 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2990 }
2991
Colin Cross98be1bb2019-12-13 20:41:13 -08002992 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002993
2994 run(t, ctx, config)
2995
2996 foo := ctx.ModuleForTests("foo", "android_common")
2997 dex := foo.Rule("r8")
2998 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2999 aligned := foo.MaybeRule("zipalign").Rule != nil
3000
3001 if uncompressedInDexJar != want {
3002 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
3003 }
3004
3005 if aligned != want {
3006 t.Errorf("want aligned %v, got %v", want, aligned)
3007 }
3008 }
3009
3010 for _, tt := range testCases {
3011 t.Run(tt.name, func(t *testing.T) {
3012 t.Run("platform", func(t *testing.T) {
3013 test(t, tt.bp, tt.uncompressedPlatform, false)
3014 })
3015 t.Run("unbundled", func(t *testing.T) {
3016 test(t, tt.bp, tt.uncompressedUnbundled, true)
3017 })
3018 })
3019 }
3020}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07003021
3022func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
3023 if expectedValue != "" {
3024 expectedFlag := "--" + flagName + " " + expectedValue
3025 if !strings.Contains(aapt2Flags, expectedFlag) {
3026 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
3027 }
3028 } else {
3029 unexpectedFlag := "--" + flagName
3030 if strings.Contains(aapt2Flags, unexpectedFlag) {
3031 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
3032 }
3033 }
3034}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003035
3036func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungca095d72020-04-09 16:15:30 -07003037 fs := map[string][]byte{
3038 "baz/res/res/values/strings.xml": nil,
3039 "bar/res/res/values/strings.xml": nil,
3040 }
3041 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003042 runtime_resource_overlay {
3043 name: "foo",
3044 certificate: "platform",
Liz Kammer7fe241f2020-05-19 16:15:25 -07003045 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003046 product_specific: true,
Jaewoong Jungca095d72020-04-09 16:15:30 -07003047 static_libs: ["bar"],
3048 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003049 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003050 }
3051
3052 runtime_resource_overlay {
3053 name: "foo_themed",
3054 certificate: "platform",
3055 product_specific: true,
3056 theme: "faza",
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003057 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003058 }
Jaewoong Jungca095d72020-04-09 16:15:30 -07003059
3060 android_library {
3061 name: "bar",
3062 resource_dirs: ["bar/res"],
3063 }
3064
3065 android_app {
3066 name: "baz",
3067 sdk_version: "current",
3068 resource_dirs: ["baz/res"],
3069 }
3070 `
3071 config := testAppConfig(nil, bp, fs)
3072 ctx := testContext()
3073 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003074
3075 m := ctx.ModuleForTests("foo", "android_common")
3076
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003077 // Check AAPT2 link flags.
3078 aapt2Flags := m.Output("package-res.apk").Args["flags"]
3079 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3080 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
3081 if len(absentFlags) > 0 {
3082 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3083 }
3084
Jaewoong Jungca095d72020-04-09 16:15:30 -07003085 // Check overlay.list output for static_libs dependency.
3086 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
3087 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
3088 if !inList(staticLibPackage, overlayList) {
3089 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
3090 }
3091
3092 // Check AAPT2 link flags for resource_libs dependency.
3093 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
3094 if !strings.Contains(aapt2Flags, resourceLibFlag) {
3095 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
3096 }
3097
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003098 // Check cert signing flag.
3099 signedApk := m.Output("signed/foo.apk")
Liz Kammer7fe241f2020-05-19 16:15:25 -07003100 lineageFlag := signedApk.Args["flags"]
3101 expectedLineageFlag := "--lineage lineage.bin"
3102 if expectedLineageFlag != lineageFlag {
3103 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
3104 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003105 signingFlag := signedApk.Args["certificates"]
3106 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
3107 if expected != signingFlag {
3108 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
3109 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003110 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3111 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003112 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
3113 if !reflect.DeepEqual(path, expectedPath) {
3114 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
3115 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003116
3117 // Check device location.
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003118 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003119 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003120 if !reflect.DeepEqual(path, expectedPath) {
3121 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3122 }
3123
3124 // A themed module has a different device location
3125 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003126 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3127 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003128 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3129 if !reflect.DeepEqual(path, expectedPath) {
3130 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3131 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003132
3133 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3134 expectedOverrides := []string{"foo"}
3135 if !reflect.DeepEqual(overrides, expectedOverrides) {
3136 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3137 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003138}
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07003139
3140func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3141 ctx, config := testJava(t, `
3142 java_defaults {
3143 name: "rro_defaults",
3144 theme: "default_theme",
3145 product_specific: true,
3146 aaptflags: ["--keep-raw-values"],
3147 }
3148
3149 runtime_resource_overlay {
3150 name: "foo_with_defaults",
3151 defaults: ["rro_defaults"],
3152 }
3153
3154 runtime_resource_overlay {
3155 name: "foo_barebones",
3156 }
3157 `)
3158
3159 //
3160 // RRO module with defaults
3161 //
3162 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3163
3164 // Check AAPT2 link flags.
3165 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3166 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3167 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3168 if len(absentFlags) > 0 {
3169 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3170 }
3171
3172 // Check device location.
3173 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3174 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3175 if !reflect.DeepEqual(path, expectedPath) {
3176 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3177 }
3178
3179 //
3180 // RRO module without defaults
3181 //
3182 m = ctx.ModuleForTests("foo_barebones", "android_common")
3183
3184 // Check AAPT2 link flags.
3185 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3186 unexpectedFlags := "--keep-raw-values"
3187 if inList(unexpectedFlags, aapt2Flags) {
3188 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3189 }
3190
3191 // Check device location.
3192 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3193 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3194 if !reflect.DeepEqual(path, expectedPath) {
3195 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3196 }
3197}
Roshan Piusb8307962020-04-27 09:42:27 -07003198
3199func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3200 ctx, _ := testJava(t, `
3201 runtime_resource_overlay {
3202 name: "foo_overlay",
3203 certificate: "platform",
3204 product_specific: true,
3205 sdk_version: "current",
3206 }
3207
3208 override_runtime_resource_overlay {
3209 name: "bar_overlay",
3210 base: "foo_overlay",
3211 package_name: "com.android.bar.overlay",
3212 target_package_name: "com.android.bar",
3213 }
3214 `)
3215
3216 expectedVariants := []struct {
3217 moduleName string
3218 variantName string
3219 apkPath string
3220 overrides []string
3221 targetVariant string
3222 packageFlag string
3223 targetPackageFlag string
3224 }{
3225 {
3226 variantName: "android_common",
3227 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3228 overrides: nil,
3229 targetVariant: "android_common",
3230 packageFlag: "",
3231 targetPackageFlag: "",
3232 },
3233 {
3234 variantName: "android_common_bar_overlay",
3235 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3236 overrides: []string{"foo_overlay"},
3237 targetVariant: "android_common_bar",
3238 packageFlag: "com.android.bar.overlay",
3239 targetPackageFlag: "com.android.bar",
3240 },
3241 }
3242 for _, expected := range expectedVariants {
3243 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3244
3245 // Check the final apk name
3246 outputs := variant.AllOutputs()
3247 expectedApkPath := buildDir + expected.apkPath
3248 found := false
3249 for _, o := range outputs {
3250 if o == expectedApkPath {
3251 found = true
3252 break
3253 }
3254 }
3255 if !found {
3256 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3257 }
3258
3259 // Check if the overrides field values are correctly aggregated.
3260 mod := variant.Module().(*RuntimeResourceOverlay)
3261 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3262 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3263 expected.overrides, mod.properties.Overrides)
3264 }
3265
3266 // Check aapt2 flags.
3267 res := variant.Output("package-res.apk")
3268 aapt2Flags := res.Args["flags"]
3269 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammerf9e5c3b2020-06-18 19:44:06 +00003270 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", "")
Roshan Piusb8307962020-04-27 09:42:27 -07003271 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3272 }
3273}