blob: 6b83cd2e15d38afe801fec41cb12c632cfcc08a5 [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 Han749dc692020-04-15 11:03:39 +0900481func TestUpdatableApps_TransitiveDepsShouldSetMinSdkVersion(t *testing.T) {
482 testJavaError(t, `module "bar".*: should support min_sdk_version\(29\)`, 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: "29",
489 static_libs: ["bar"],
490 }
491
492 java_library {
493 name: "bar",
494 sdk_version: "current",
495 }
496 `)
497}
498
Jooyung Hanaf7f91f2020-04-29 14:01:06 +0900499func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
500 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
501 android_app {
502 name: "foo",
503 srcs: ["a.java"],
504 updatable: true,
505 sdk_version: "current",
506 min_sdk_version: "current",
507 jni_libs: ["libjni"],
508 }
509
510 cc_library {
511 name: "libjni",
512 stl: "none",
513 system_shared_libs: [],
514 sdk_version: "current",
515 }
516 `)
517}
518
519func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
520 bp := cc.GatherRequiredDepsForTest(android.Android) + `
521 android_app {
522 name: "foo",
523 srcs: ["a.java"],
524 updatable: true,
525 sdk_version: "current",
526 min_sdk_version: "29",
527 jni_libs: ["libjni"],
528 }
529
530 cc_library {
531 name: "libjni",
532 stl: "none",
533 system_shared_libs: [],
534 sdk_version: "29",
535 }
536
537 ndk_prebuilt_object {
538 name: "ndk_crtbegin_so.29",
539 sdk_version: "29",
540 }
541
542 ndk_prebuilt_object {
543 name: "ndk_crtend_so.29",
544 sdk_version: "29",
545 }
546 `
547 fs := map[string][]byte{
548 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
549 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
550 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
551 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
552 }
553
554 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
555
556 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
557 var crtbeginFound, crtendFound bool
558 for _, input := range inputs {
559 switch input.String() {
560 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o":
561 crtbeginFound = true
562 case "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o":
563 crtendFound = true
564 }
565 }
566 if !crtbeginFound || !crtendFound {
567 t.Error("should link with ndk_crtbegin_so.29 and ndk_crtend_so.29")
568 }
569}
570
571func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
572 bp := cc.GatherRequiredDepsForTest(android.Android) + `
573 android_app {
574 name: "foo",
575 srcs: ["a.java"],
576 updatable: true,
577 sdk_version: "current",
578 min_sdk_version: "29", // this APK should support 29
579 jni_libs: ["libjni"],
580 }
581
582 cc_library {
583 name: "libjni",
584 stl: "none",
585 sdk_version: "current",
586 }
587 `
588 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
589}
590
591func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
592 bp := cc.GatherRequiredDepsForTest(android.Android) + `
593 android_app {
594 name: "foo",
595 srcs: ["a.java"],
596 updatable: true,
597 sdk_version: "current",
598 min_sdk_version: "29", // this APK should support 29
599 jni_libs: ["libjni"],
600 }
601
602 cc_library {
603 name: "libjni",
604 stl: "none",
605 shared_libs: ["libbar"],
606 system_shared_libs: [],
607 sdk_version: "27",
608 }
609
610 cc_library {
611 name: "libbar",
612 stl: "none",
613 system_shared_libs: [],
614 sdk_version: "current",
615 }
616 `
617 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
618}
619
Colin Cross0ddae7f2019-02-07 15:30:01 -0800620func TestResourceDirs(t *testing.T) {
621 testCases := []struct {
622 name string
623 prop string
624 resources []string
625 }{
626 {
627 name: "no resource_dirs",
628 prop: "",
629 resources: []string{"res/res/values/strings.xml"},
630 },
631 {
632 name: "resource_dirs",
633 prop: `resource_dirs: ["res"]`,
634 resources: []string{"res/res/values/strings.xml"},
635 },
636 {
637 name: "empty resource_dirs",
638 prop: `resource_dirs: []`,
639 resources: nil,
640 },
641 }
642
643 fs := map[string][]byte{
644 "res/res/values/strings.xml": nil,
645 }
646
647 bp := `
648 android_app {
649 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900650 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800651 %s
652 }
653 `
654
655 for _, testCase := range testCases {
656 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800657 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
658 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800659 run(t, ctx, config)
660
661 module := ctx.ModuleForTests("foo", "android_common")
662 resourceList := module.MaybeOutput("aapt2/res.list")
663
664 var resources []string
665 if resourceList.Rule != nil {
666 for _, compiledResource := range resourceList.Inputs.Strings() {
667 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
668 }
669 }
670
671 if !reflect.DeepEqual(resources, testCase.resources) {
672 t.Errorf("expected resource files %q, got %q",
673 testCase.resources, resources)
674 }
675 })
676 }
677}
678
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800679func TestLibraryAssets(t *testing.T) {
680 bp := `
681 android_app {
682 name: "foo",
683 sdk_version: "current",
684 static_libs: ["lib1", "lib2", "lib3"],
685 }
686
687 android_library {
688 name: "lib1",
689 sdk_version: "current",
690 asset_dirs: ["assets_a"],
691 }
692
693 android_library {
694 name: "lib2",
695 sdk_version: "current",
696 }
697
698 android_library {
699 name: "lib3",
700 sdk_version: "current",
701 static_libs: ["lib4"],
702 }
703
704 android_library {
705 name: "lib4",
706 sdk_version: "current",
707 asset_dirs: ["assets_b"],
708 }
709 `
710
711 testCases := []struct {
712 name string
713 assetFlag string
714 assetPackages []string
715 }{
716 {
717 name: "foo",
718 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
719 assetPackages: []string{
720 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
721 buildDir + "/.intermediates/lib1/android_common/assets.zip",
722 buildDir + "/.intermediates/lib3/android_common/assets.zip",
723 },
724 },
725 {
726 name: "lib1",
727 assetFlag: "-A assets_a",
728 },
729 {
730 name: "lib2",
731 },
732 {
733 name: "lib3",
734 assetPackages: []string{
735 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
736 buildDir + "/.intermediates/lib4/android_common/assets.zip",
737 },
738 },
739 {
740 name: "lib4",
741 assetFlag: "-A assets_b",
742 },
743 }
744 ctx := testApp(t, bp)
745
746 for _, test := range testCases {
747 t.Run(test.name, func(t *testing.T) {
748 m := ctx.ModuleForTests(test.name, "android_common")
749
750 // Check asset flag in aapt2 link flags
751 var aapt2link android.TestingBuildParams
752 if len(test.assetPackages) > 0 {
753 aapt2link = m.Output("aapt2/package-res.apk")
754 } else {
755 aapt2link = m.Output("package-res.apk")
756 }
757 aapt2Flags := aapt2link.Args["flags"]
758 if test.assetFlag != "" {
759 if !strings.Contains(aapt2Flags, test.assetFlag) {
760 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
761 }
762 } else {
763 if strings.Contains(aapt2Flags, " -A ") {
764 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
765 }
766 }
767
768 // Check asset merge rule.
769 if len(test.assetPackages) > 0 {
770 mergeAssets := m.Output("package-res.apk")
771 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
772 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
773 mergeAssets.Inputs.Strings(), test.assetPackages)
774 }
775 }
776 })
777 }
778}
779
Colin Crossbec85302019-02-13 13:15:46 -0800780func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800781 testCases := []struct {
782 name string
783 enforceRROTargets []string
784 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800785 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800786 overlayFiles map[string][]string
787 rroDirs map[string][]string
788 }{
789 {
790 name: "no RRO",
791 enforceRROTargets: nil,
792 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800793 resourceFiles: map[string][]string{
794 "foo": nil,
795 "bar": {"bar/res/res/values/strings.xml"},
796 "lib": nil,
797 "lib2": {"lib2/res/res/values/strings.xml"},
798 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800799 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800800 "foo": {
801 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800802 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000803 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800804 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800805 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
806 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000807 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800808 },
Colin Crossbec85302019-02-13 13:15:46 -0800809 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800810 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
811 "device/vendor/blah/overlay/bar/res/values/strings.xml",
812 },
Colin Crossbec85302019-02-13 13:15:46 -0800813 "lib": {
814 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
815 "lib/res/res/values/strings.xml",
816 "device/vendor/blah/overlay/lib/res/values/strings.xml",
817 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800818 },
819 rroDirs: map[string][]string{
820 "foo": nil,
821 "bar": nil,
822 },
823 },
824 {
825 name: "enforce RRO on foo",
826 enforceRROTargets: []string{"foo"},
827 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800828 resourceFiles: map[string][]string{
829 "foo": nil,
830 "bar": {"bar/res/res/values/strings.xml"},
831 "lib": nil,
832 "lib2": {"lib2/res/res/values/strings.xml"},
833 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800834 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800835 "foo": {
836 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800837 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000838 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800839 "foo/res/res/values/strings.xml",
840 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
841 },
Colin Crossbec85302019-02-13 13:15:46 -0800842 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800843 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
844 "device/vendor/blah/overlay/bar/res/values/strings.xml",
845 },
Colin Crossbec85302019-02-13 13:15:46 -0800846 "lib": {
847 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
848 "lib/res/res/values/strings.xml",
849 "device/vendor/blah/overlay/lib/res/values/strings.xml",
850 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800851 },
Colin Crossc1c37552019-01-31 11:42:41 -0800852
Colin Cross5c4791c2019-02-01 11:44:44 -0800853 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800854 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000855 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800856 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
857 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000858 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800859 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800860 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800861 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800862 },
863 },
864 {
865 name: "enforce RRO on all",
866 enforceRROTargets: []string{"*"},
867 enforceRROExcludedOverlays: []string{
868 // Excluding specific apps/res directories also allowed.
869 "device/vendor/blah/static_overlay/foo",
870 "device/vendor/blah/static_overlay/bar/res",
871 },
Colin Crossbec85302019-02-13 13:15:46 -0800872 resourceFiles: map[string][]string{
873 "foo": nil,
874 "bar": {"bar/res/res/values/strings.xml"},
875 "lib": nil,
876 "lib2": {"lib2/res/res/values/strings.xml"},
877 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800878 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800879 "foo": {
880 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800881 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000882 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800883 "foo/res/res/values/strings.xml",
884 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
885 },
Colin Crossbec85302019-02-13 13:15:46 -0800886 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
887 "lib": {
888 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
889 "lib/res/res/values/strings.xml",
890 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800891 },
892 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800893 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000894 "device:device/vendor/blah/overlay/foo/res",
895 "product:product/vendor/blah/overlay/foo/res",
896 // Lib dep comes after the direct deps
897 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800898 },
Anton Hansson53c88442019-03-18 15:53:16 +0000899 "bar": {"device:device/vendor/blah/overlay/bar/res"},
900 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800901 },
902 },
903 }
904
Anton Hansson53c88442019-03-18 15:53:16 +0000905 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800906 "device/vendor/blah/overlay",
907 "device/vendor/blah/overlay2",
908 "device/vendor/blah/static_overlay",
909 }
910
Anton Hansson53c88442019-03-18 15:53:16 +0000911 productResourceOverlays := []string{
912 "product/vendor/blah/overlay",
913 }
914
Colin Cross890ff552017-11-30 20:13:19 -0800915 fs := map[string][]byte{
916 "foo/res/res/values/strings.xml": nil,
917 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800918 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800919 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800920 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
921 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800922 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800923 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
924 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
925 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000926 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800927 }
928
929 bp := `
930 android_app {
931 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900932 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800933 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000934 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800935 }
936
937 android_app {
938 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900939 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800940 resource_dirs: ["bar/res"],
941 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800942
943 android_library {
944 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900945 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800946 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800947 static_libs: ["lib2"],
948 }
949
950 android_library {
951 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900952 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800953 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800954 }
Anton Hansson53c88442019-03-18 15:53:16 +0000955
956 // This library has the same resources as lib (should not lead to dupe RROs)
957 android_library {
958 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900959 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000960 resource_dirs: ["lib/res"]
961 }
Colin Cross890ff552017-11-30 20:13:19 -0800962 `
963
Colin Cross5c4791c2019-02-01 11:44:44 -0800964 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800965 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800966 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000967 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
968 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800969 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800970 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800971 }
972 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800973 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800974 }
975
Colin Cross98be1bb2019-12-13 20:41:13 -0800976 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800977 run(t, ctx, config)
978
Colin Crossbec85302019-02-13 13:15:46 -0800979 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
980 for _, o := range list {
981 res := module.MaybeOutput(o)
982 if res.Rule != nil {
983 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
984 // verify the inputs to the .arsc.flat rule.
985 files = append(files, res.Inputs.Strings()...)
986 } else {
987 // Otherwise, verify the full path to the output of the other module
988 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000989 }
Colin Cross890ff552017-11-30 20:13:19 -0800990 }
Colin Crossbec85302019-02-13 13:15:46 -0800991 return files
Colin Cross890ff552017-11-30 20:13:19 -0800992 }
993
Colin Crossbec85302019-02-13 13:15:46 -0800994 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
995 module := ctx.ModuleForTests(moduleName, "android_common")
996 resourceList := module.MaybeOutput("aapt2/res.list")
997 if resourceList.Rule != nil {
998 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000999 }
Colin Crossbec85302019-02-13 13:15:46 -08001000 overlayList := module.MaybeOutput("aapt2/overlay.list")
1001 if overlayList.Rule != nil {
1002 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
1003 }
1004
Anton Hansson53c88442019-03-18 15:53:16 +00001005 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
1006 var prefix string
1007 if d.overlayType == device {
1008 prefix = "device:"
1009 } else if d.overlayType == product {
1010 prefix = "product:"
1011 } else {
1012 t.Fatalf("Unexpected overlayType %d", d.overlayType)
1013 }
1014 rroDirs = append(rroDirs, prefix+d.path.String())
1015 }
Colin Crossbec85302019-02-13 13:15:46 -08001016
1017 return resourceFiles, overlayFiles, rroDirs
1018 }
1019
1020 modules := []string{"foo", "bar", "lib", "lib2"}
1021 for _, module := range modules {
1022 resourceFiles, overlayFiles, rroDirs := getResources(module)
1023
1024 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
1025 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
1026 module, testCase.resourceFiles[module], resourceFiles)
1027 }
1028 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
1029 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
1030 module, testCase.overlayFiles[module], overlayFiles)
1031 }
1032 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +00001033 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -08001034 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +00001035 }
Colin Cross890ff552017-11-30 20:13:19 -08001036 }
Colin Cross890ff552017-11-30 20:13:19 -08001037 })
1038 }
1039}
Colin Crossd09b0b62018-04-18 11:06:47 -07001040
1041func TestAppSdkVersion(t *testing.T) {
1042 testCases := []struct {
1043 name string
1044 sdkVersion string
1045 platformSdkInt int
1046 platformSdkCodename string
1047 platformSdkFinal bool
1048 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +09001049 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -07001050 }{
1051 {
1052 name: "current final SDK",
1053 sdkVersion: "current",
1054 platformSdkInt: 27,
1055 platformSdkCodename: "REL",
1056 platformSdkFinal: true,
1057 expectedMinSdkVersion: "27",
1058 },
1059 {
1060 name: "current non-final SDK",
1061 sdkVersion: "current",
1062 platformSdkInt: 27,
1063 platformSdkCodename: "OMR1",
1064 platformSdkFinal: false,
1065 expectedMinSdkVersion: "OMR1",
1066 },
1067 {
1068 name: "default final SDK",
1069 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001070 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001071 platformSdkInt: 27,
1072 platformSdkCodename: "REL",
1073 platformSdkFinal: true,
1074 expectedMinSdkVersion: "27",
1075 },
1076 {
1077 name: "default non-final SDK",
1078 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001079 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001080 platformSdkInt: 27,
1081 platformSdkCodename: "OMR1",
1082 platformSdkFinal: false,
1083 expectedMinSdkVersion: "OMR1",
1084 },
1085 {
1086 name: "14",
1087 sdkVersion: "14",
1088 expectedMinSdkVersion: "14",
1089 },
1090 }
1091
1092 for _, moduleType := range []string{"android_app", "android_library"} {
1093 for _, test := range testCases {
1094 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001095 platformApiProp := ""
1096 if test.platformApis {
1097 platformApiProp = "platform_apis: true,"
1098 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001099 bp := fmt.Sprintf(`%s {
1100 name: "foo",
1101 srcs: ["a.java"],
1102 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001103 %s
1104 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001105
Colin Cross98be1bb2019-12-13 20:41:13 -08001106 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001107 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1108 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1109 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1110
Colin Cross98be1bb2019-12-13 20:41:13 -08001111 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -07001112
1113 run(t, ctx, config)
1114
1115 foo := ctx.ModuleForTests("foo", "android_common")
1116 link := foo.Output("package-res.apk")
1117 linkFlags := strings.Split(link.Args["flags"], " ")
1118 min := android.IndexList("--min-sdk-version", linkFlags)
1119 target := android.IndexList("--target-sdk-version", linkFlags)
1120
1121 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1122 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1123 }
1124
1125 gotMinSdkVersion := linkFlags[min+1]
1126 gotTargetSdkVersion := linkFlags[target+1]
1127
1128 if gotMinSdkVersion != test.expectedMinSdkVersion {
1129 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1130 test.expectedMinSdkVersion, gotMinSdkVersion)
1131 }
1132
1133 if gotTargetSdkVersion != test.expectedMinSdkVersion {
1134 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1135 test.expectedMinSdkVersion, gotTargetSdkVersion)
1136 }
1137 })
1138 }
1139 }
1140}
Colin Crossa4f08812018-10-02 22:03:40 -07001141
Paul Duffin50c217c2019-06-12 13:25:22 +01001142func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001143 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001144 cc_library {
1145 name: "libjni",
1146 system_shared_libs: [],
Colin Cross01fd7cc2020-02-19 16:54:04 -08001147 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001148 stl: "none",
1149 }
1150
1151 android_test {
1152 name: "test",
1153 sdk_version: "core_platform",
1154 jni_libs: ["libjni"],
1155 }
1156
1157 android_test {
1158 name: "test_first",
1159 sdk_version: "core_platform",
1160 compile_multilib: "first",
1161 jni_libs: ["libjni"],
1162 }
1163
1164 android_test {
1165 name: "test_both",
1166 sdk_version: "core_platform",
1167 compile_multilib: "both",
1168 jni_libs: ["libjni"],
1169 }
1170
1171 android_test {
1172 name: "test_32",
1173 sdk_version: "core_platform",
1174 compile_multilib: "32",
1175 jni_libs: ["libjni"],
1176 }
1177
1178 android_test {
1179 name: "test_64",
1180 sdk_version: "core_platform",
1181 compile_multilib: "64",
1182 jni_libs: ["libjni"],
1183 }
1184 `)
1185
1186 testCases := []struct {
1187 name string
1188 abis []string
1189 }{
1190 {"test", []string{"arm64-v8a"}},
1191 {"test_first", []string{"arm64-v8a"}},
1192 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1193 {"test_32", []string{"armeabi-v7a"}},
1194 {"test_64", []string{"arm64-v8a"}},
1195 }
1196
1197 for _, test := range testCases {
1198 t.Run(test.name, func(t *testing.T) {
1199 app := ctx.ModuleForTests(test.name, "android_common")
1200 jniLibZip := app.Output("jnilibs.zip")
1201 var abis []string
1202 args := strings.Fields(jniLibZip.Args["jarArgs"])
1203 for i := 0; i < len(args); i++ {
1204 if args[i] == "-P" {
1205 abis = append(abis, filepath.Base(args[i+1]))
1206 i++
1207 }
1208 }
1209 if !reflect.DeepEqual(abis, test.abis) {
1210 t.Errorf("want abis %v, got %v", test.abis, abis)
1211 }
1212 })
1213 }
1214}
1215
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001216func TestAppSdkVersionByPartition(t *testing.T) {
1217 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1218 android_app {
1219 name: "foo",
1220 srcs: ["a.java"],
1221 vendor: true,
1222 platform_apis: true,
1223 }
1224 `)
1225
1226 testJava(t, `
1227 android_app {
1228 name: "bar",
1229 srcs: ["b.java"],
1230 platform_apis: true,
1231 }
1232 `)
1233
1234 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001235 bp := `
1236 android_app {
1237 name: "foo",
1238 srcs: ["a.java"],
1239 product_specific: true,
1240 platform_apis: true,
1241 }
1242 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001243
1244 config := testAppConfig(nil, bp, nil)
1245 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001246 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001247 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 +09001248 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001249 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001250 }
1251 }
1252}
1253
Paul Duffin50c217c2019-06-12 13:25:22 +01001254func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001255 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001256 cc_library {
1257 name: "libjni",
1258 system_shared_libs: [],
1259 stl: "none",
Colin Cross1c93c292020-02-15 10:38:00 -08001260 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001261 }
1262
1263 android_app {
1264 name: "app",
1265 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001266 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001267 }
1268
1269 android_app {
1270 name: "app_noembed",
1271 jni_libs: ["libjni"],
1272 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001273 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001274 }
1275
1276 android_app {
1277 name: "app_embed",
1278 jni_libs: ["libjni"],
1279 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001280 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001281 }
1282
1283 android_test {
1284 name: "test",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001285 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001286 jni_libs: ["libjni"],
1287 }
1288
1289 android_test {
1290 name: "test_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001291 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001292 jni_libs: ["libjni"],
1293 use_embedded_native_libs: false,
1294 }
1295
1296 android_test_helper_app {
1297 name: "test_helper",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001298 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001299 jni_libs: ["libjni"],
1300 }
1301
1302 android_test_helper_app {
1303 name: "test_helper_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001304 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001305 jni_libs: ["libjni"],
1306 use_embedded_native_libs: false,
1307 }
1308 `)
1309
1310 testCases := []struct {
1311 name string
1312 packaged bool
1313 compressed bool
1314 }{
1315 {"app", false, false},
1316 {"app_noembed", false, false},
1317 {"app_embed", true, false},
1318 {"test", true, false},
1319 {"test_noembed", true, true},
1320 {"test_helper", true, false},
1321 {"test_helper_noembed", true, true},
1322 }
1323
1324 for _, test := range testCases {
1325 t.Run(test.name, func(t *testing.T) {
1326 app := ctx.ModuleForTests(test.name, "android_common")
1327 jniLibZip := app.MaybeOutput("jnilibs.zip")
1328 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1329 t.Errorf("expected jni packaged %v, got %v", w, g)
1330 }
1331
1332 if jniLibZip.Rule != nil {
1333 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1334 t.Errorf("expected jni compressed %v, got %v", w, g)
1335 }
Colin Cross01fd7cc2020-02-19 16:54:04 -08001336
1337 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1338 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1339 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001340 }
1341 })
1342 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001343}
1344
Colin Cross1dd9c442020-05-08 11:20:24 -07001345func TestJNISDK(t *testing.T) {
1346 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1347 cc_library {
1348 name: "libjni",
1349 system_shared_libs: [],
1350 stl: "none",
1351 sdk_version: "current",
1352 }
1353
1354 android_test {
1355 name: "app_platform",
1356 jni_libs: ["libjni"],
1357 platform_apis: true,
1358 }
1359
1360 android_test {
1361 name: "app_sdk",
1362 jni_libs: ["libjni"],
1363 sdk_version: "current",
1364 }
1365
1366 android_test {
1367 name: "app_force_platform",
1368 jni_libs: ["libjni"],
1369 sdk_version: "current",
1370 jni_uses_platform_apis: true,
1371 }
1372
1373 android_test {
1374 name: "app_force_sdk",
1375 jni_libs: ["libjni"],
1376 platform_apis: true,
1377 jni_uses_sdk_apis: true,
1378 }
Colin Crosseb032962020-05-13 11:05:02 -07001379
1380 cc_library {
1381 name: "libvendorjni",
1382 system_shared_libs: [],
1383 stl: "none",
1384 vendor: true,
1385 }
1386
1387 android_test {
1388 name: "app_vendor",
1389 jni_libs: ["libvendorjni"],
1390 sdk_version: "current",
1391 vendor: true,
1392 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001393 `)
1394
1395 testCases := []struct {
Colin Crosseb032962020-05-13 11:05:02 -07001396 name string
1397 sdkJNI bool
1398 vendorJNI bool
Colin Cross1dd9c442020-05-08 11:20:24 -07001399 }{
Colin Crosseb032962020-05-13 11:05:02 -07001400 {name: "app_platform"},
1401 {name: "app_sdk", sdkJNI: true},
1402 {name: "app_force_platform"},
1403 {name: "app_force_sdk", sdkJNI: true},
1404 {name: "app_vendor", vendorJNI: true},
Colin Cross1dd9c442020-05-08 11:20:24 -07001405 }
1406
Colin Crosseb032962020-05-13 11:05:02 -07001407 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1408 Output("libjni.so").Output.String()
1409 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1410 Output("libjni.so").Output.String()
1411 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1412 Output("libvendorjni.so").Output.String()
1413
Colin Cross1dd9c442020-05-08 11:20:24 -07001414 for _, test := range testCases {
1415 t.Run(test.name, func(t *testing.T) {
1416 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross1dd9c442020-05-08 11:20:24 -07001417
1418 jniLibZip := app.MaybeOutput("jnilibs.zip")
1419 if len(jniLibZip.Implicits) != 1 {
1420 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1421 }
1422 gotJNI := jniLibZip.Implicits[0].String()
1423
1424 if test.sdkJNI {
1425 if gotJNI != sdkJNI {
1426 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1427 }
Colin Crosseb032962020-05-13 11:05:02 -07001428 } else if test.vendorJNI {
1429 if gotJNI != vendorJNI {
1430 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1431 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001432 } else {
1433 if gotJNI != platformJNI {
1434 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1435 }
1436 }
1437 })
1438 }
1439
1440 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1441 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1442 android_test {
1443 name: "app_platform",
1444 platform_apis: true,
1445 jni_uses_platform_apis: true,
1446 }
1447 `)
1448 })
1449
1450 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1451 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1452 android_test {
1453 name: "app_sdk",
1454 sdk_version: "current",
1455 jni_uses_sdk_apis: true,
1456 }
1457 `)
1458 })
1459
1460}
1461
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001462func TestCertificates(t *testing.T) {
1463 testCases := []struct {
1464 name string
1465 bp string
1466 certificateOverride string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001467 expectedLineage string
1468 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001469 }{
1470 {
1471 name: "default",
1472 bp: `
1473 android_app {
1474 name: "foo",
1475 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001476 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001477 }
1478 `,
1479 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001480 expectedLineage: "",
1481 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001482 },
1483 {
1484 name: "module certificate property",
1485 bp: `
1486 android_app {
1487 name: "foo",
1488 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001489 certificate: ":new_certificate",
1490 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001491 }
1492
1493 android_app_certificate {
1494 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001495 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001496 }
1497 `,
1498 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001499 expectedLineage: "",
1500 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001501 },
1502 {
1503 name: "path certificate property",
1504 bp: `
1505 android_app {
1506 name: "foo",
1507 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001508 certificate: "expiredkey",
1509 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001510 }
1511 `,
1512 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001513 expectedLineage: "",
1514 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001515 },
1516 {
1517 name: "certificate overrides",
1518 bp: `
1519 android_app {
1520 name: "foo",
1521 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001522 certificate: "expiredkey",
1523 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001524 }
1525
1526 android_app_certificate {
1527 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001528 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001529 }
1530 `,
1531 certificateOverride: "foo:new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001532 expectedLineage: "",
1533 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1534 },
1535 {
1536 name: "certificate lineage",
1537 bp: `
1538 android_app {
1539 name: "foo",
1540 srcs: ["a.java"],
1541 certificate: ":new_certificate",
1542 lineage: "lineage.bin",
1543 sdk_version: "current",
1544 }
1545
1546 android_app_certificate {
1547 name: "new_certificate",
1548 certificate: "cert/new_cert",
1549 }
1550 `,
1551 certificateOverride: "",
1552 expectedLineage: "--lineage lineage.bin",
1553 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001554 },
1555 }
1556
1557 for _, test := range testCases {
1558 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001559 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001560 if test.certificateOverride != "" {
1561 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1562 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001563 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001564
1565 run(t, ctx, config)
1566 foo := ctx.ModuleForTests("foo", "android_common")
1567
1568 signapk := foo.Output("foo.apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001569 signCertificateFlags := signapk.Args["certificates"]
1570 if test.expectedCertificate != signCertificateFlags {
1571 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1572 }
1573
1574 signFlags := signapk.Args["flags"]
1575 if test.expectedLineage != signFlags {
1576 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001577 }
1578 })
1579 }
1580}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001581
Songchun Fan688de9a2020-03-24 20:32:24 -07001582func TestRequestV4SigningFlag(t *testing.T) {
1583 testCases := []struct {
1584 name string
1585 bp string
1586 expected string
1587 }{
1588 {
1589 name: "default",
1590 bp: `
1591 android_app {
1592 name: "foo",
1593 srcs: ["a.java"],
1594 sdk_version: "current",
1595 }
1596 `,
1597 expected: "",
1598 },
1599 {
1600 name: "default",
1601 bp: `
1602 android_app {
1603 name: "foo",
1604 srcs: ["a.java"],
1605 sdk_version: "current",
1606 v4_signature: false,
1607 }
1608 `,
1609 expected: "",
1610 },
1611 {
1612 name: "module certificate property",
1613 bp: `
1614 android_app {
1615 name: "foo",
1616 srcs: ["a.java"],
1617 sdk_version: "current",
1618 v4_signature: true,
1619 }
1620 `,
1621 expected: "--enable-v4",
1622 },
1623 }
1624
1625 for _, test := range testCases {
1626 t.Run(test.name, func(t *testing.T) {
1627 config := testAppConfig(nil, test.bp, nil)
1628 ctx := testContext()
1629
1630 run(t, ctx, config)
1631 foo := ctx.ModuleForTests("foo", "android_common")
1632
1633 signapk := foo.Output("foo.apk")
1634 signFlags := signapk.Args["flags"]
1635 if test.expected != signFlags {
1636 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1637 }
1638 })
1639 }
1640}
1641
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001642func TestPackageNameOverride(t *testing.T) {
1643 testCases := []struct {
1644 name string
1645 bp string
1646 packageNameOverride string
1647 expected []string
1648 }{
1649 {
1650 name: "default",
1651 bp: `
1652 android_app {
1653 name: "foo",
1654 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001655 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001656 }
1657 `,
1658 packageNameOverride: "",
1659 expected: []string{
1660 buildDir + "/.intermediates/foo/android_common/foo.apk",
1661 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1662 },
1663 },
1664 {
1665 name: "overridden",
1666 bp: `
1667 android_app {
1668 name: "foo",
1669 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001670 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001671 }
1672 `,
1673 packageNameOverride: "foo:bar",
1674 expected: []string{
1675 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001676 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001677 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1678 },
1679 },
1680 }
1681
1682 for _, test := range testCases {
1683 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001684 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001685 if test.packageNameOverride != "" {
1686 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1687 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001688 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001689
1690 run(t, ctx, config)
1691 foo := ctx.ModuleForTests("foo", "android_common")
1692
1693 outputs := foo.AllOutputs()
1694 outputMap := make(map[string]bool)
1695 for _, o := range outputs {
1696 outputMap[o] = true
1697 }
1698 for _, e := range test.expected {
1699 if _, exist := outputMap[e]; !exist {
1700 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1701 }
1702 }
1703 })
1704 }
1705}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001706
1707func TestInstrumentationTargetOverridden(t *testing.T) {
1708 bp := `
1709 android_app {
1710 name: "foo",
1711 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001712 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001713 }
1714
1715 android_test {
1716 name: "bar",
1717 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001718 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001719 }
1720 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001721 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001722 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001723 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001724
1725 run(t, ctx, config)
1726
1727 bar := ctx.ModuleForTests("bar", "android_common")
1728 res := bar.Output("package-res.apk")
1729 aapt2Flags := res.Args["flags"]
1730 e := "--rename-instrumentation-target-package org.dandroid.bp"
1731 if !strings.Contains(aapt2Flags, e) {
1732 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1733 }
1734}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001735
1736func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001737 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001738 android_app {
1739 name: "foo",
1740 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001741 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001742 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001743 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001744 }
1745
1746 override_android_app {
1747 name: "bar",
1748 base: "foo",
1749 certificate: ":new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001750 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001751 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001752 }
1753
1754 android_app_certificate {
1755 name: "new_certificate",
1756 certificate: "cert/new_cert",
1757 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001758
1759 override_android_app {
1760 name: "baz",
1761 base: "foo",
1762 package_name: "org.dandroid.bp",
1763 }
Liz Kammer9f9fd022020-06-18 19:44:06 +00001764
1765 override_android_app {
1766 name: "baz_no_rename_resources",
1767 base: "foo",
1768 package_name: "org.dandroid.bp",
1769 rename_resources_package: false,
1770 }
1771
1772 android_app {
1773 name: "foo_no_rename_resources",
1774 srcs: ["a.java"],
1775 certificate: "expiredkey",
1776 overrides: ["qux"],
1777 rename_resources_package: false,
1778 sdk_version: "current",
1779 }
1780
1781 override_android_app {
1782 name: "baz_base_no_rename_resources",
1783 base: "foo_no_rename_resources",
1784 package_name: "org.dandroid.bp",
1785 }
1786
1787 override_android_app {
1788 name: "baz_override_base_rename_resources",
1789 base: "foo_no_rename_resources",
1790 package_name: "org.dandroid.bp",
1791 rename_resources_package: true,
1792 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001793 `)
1794
1795 expectedVariants := []struct {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001796 name string
1797 moduleName string
1798 variantName string
1799 apkName string
1800 apkPath string
1801 certFlag string
1802 lineageFlag string
1803 overrides []string
1804 packageFlag string
1805 renameResources bool
1806 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001807 }{
1808 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001809 name: "foo",
1810 moduleName: "foo",
1811 variantName: "android_common",
1812 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
1813 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1814 lineageFlag: "",
1815 overrides: []string{"qux"},
1816 packageFlag: "",
1817 renameResources: false,
1818 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001819 },
1820 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001821 name: "foo",
1822 moduleName: "bar",
1823 variantName: "android_common_bar",
1824 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
1825 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1826 lineageFlag: "--lineage lineage.bin",
1827 overrides: []string{"qux", "foo"},
1828 packageFlag: "",
1829 renameResources: false,
1830 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001831 },
1832 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001833 name: "foo",
1834 moduleName: "baz",
1835 variantName: "android_common_baz",
1836 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
1837 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1838 lineageFlag: "",
1839 overrides: []string{"qux", "foo"},
1840 packageFlag: "org.dandroid.bp",
1841 renameResources: true,
1842 logging_parent: "",
1843 },
1844 {
1845 name: "foo",
1846 moduleName: "baz_no_rename_resources",
1847 variantName: "android_common_baz_no_rename_resources",
1848 apkPath: "/target/product/test_device/system/app/baz_no_rename_resources/baz_no_rename_resources.apk",
1849 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1850 lineageFlag: "",
1851 overrides: []string{"qux", "foo"},
1852 packageFlag: "org.dandroid.bp",
1853 renameResources: false,
1854 logging_parent: "",
1855 },
1856 {
1857 name: "foo_no_rename_resources",
1858 moduleName: "baz_base_no_rename_resources",
1859 variantName: "android_common_baz_base_no_rename_resources",
1860 apkPath: "/target/product/test_device/system/app/baz_base_no_rename_resources/baz_base_no_rename_resources.apk",
1861 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1862 lineageFlag: "",
1863 overrides: []string{"qux", "foo_no_rename_resources"},
1864 packageFlag: "org.dandroid.bp",
1865 renameResources: false,
1866 logging_parent: "",
1867 },
1868 {
1869 name: "foo_no_rename_resources",
1870 moduleName: "baz_override_base_rename_resources",
1871 variantName: "android_common_baz_override_base_rename_resources",
1872 apkPath: "/target/product/test_device/system/app/baz_override_base_rename_resources/baz_override_base_rename_resources.apk",
1873 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1874 lineageFlag: "",
1875 overrides: []string{"qux", "foo_no_rename_resources"},
1876 packageFlag: "org.dandroid.bp",
1877 renameResources: true,
1878 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001879 },
1880 }
1881 for _, expected := range expectedVariants {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001882 variant := ctx.ModuleForTests(expected.name, expected.variantName)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001883
1884 // Check the final apk name
1885 outputs := variant.AllOutputs()
1886 expectedApkPath := buildDir + expected.apkPath
1887 found := false
1888 for _, o := range outputs {
1889 if o == expectedApkPath {
1890 found = true
1891 break
1892 }
1893 }
1894 if !found {
1895 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1896 }
1897
1898 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001899 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001900 certFlag := signapk.Args["certificates"]
1901 if expected.certFlag != certFlag {
1902 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1903 }
1904
1905 // Check the lineage flags
1906 lineageFlag := signapk.Args["flags"]
1907 if expected.lineageFlag != lineageFlag {
1908 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001909 }
1910
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001911 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001912 mod := variant.Module().(*AndroidApp)
1913 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1914 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1915 expected.overrides, mod.appProperties.Overrides)
1916 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001917
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001918 // Test Overridable property: Logging_parent
1919 logging_parent := mod.aapt.LoggingParent
1920 if expected.logging_parent != logging_parent {
1921 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1922 expected.logging_parent, logging_parent)
1923 }
1924
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001925 // Check the package renaming flag, if exists.
1926 res := variant.Output("package-res.apk")
1927 aapt2Flags := res.Args["flags"]
Liz Kammer9f9fd022020-06-18 19:44:06 +00001928 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1929 expectedPackage := expected.packageFlag
1930 if !expected.renameResources {
1931 expectedPackage = ""
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001932 }
Liz Kammer9f9fd022020-06-18 19:44:06 +00001933 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expectedPackage)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001934 }
1935}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001936
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001937func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001938 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001939 android_app {
1940 name: "foo",
1941 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001942 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001943 }
1944
1945 override_android_app {
1946 name: "bar",
1947 base: "foo",
1948 package_name: "org.dandroid.bp",
1949 }
1950
1951 android_test {
1952 name: "baz",
1953 srcs: ["b.java"],
1954 instrumentation_for: "foo",
1955 }
1956
1957 android_test {
1958 name: "qux",
1959 srcs: ["b.java"],
1960 instrumentation_for: "bar",
1961 }
1962 `)
1963
1964 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1965 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1966 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1967 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1968 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1969 }
1970
1971 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1972 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1973 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1974 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1975 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1976 }
1977}
1978
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001979func TestOverrideAndroidTest(t *testing.T) {
1980 ctx, _ := testJava(t, `
1981 android_app {
1982 name: "foo",
1983 srcs: ["a.java"],
1984 package_name: "com.android.foo",
1985 sdk_version: "current",
1986 }
1987
1988 override_android_app {
1989 name: "bar",
1990 base: "foo",
1991 package_name: "com.android.bar",
1992 }
1993
1994 android_test {
1995 name: "foo_test",
1996 srcs: ["b.java"],
1997 instrumentation_for: "foo",
1998 }
1999
2000 override_android_test {
2001 name: "bar_test",
2002 base: "foo_test",
2003 package_name: "com.android.bar.test",
2004 instrumentation_for: "bar",
2005 instrumentation_target_package: "com.android.bar",
2006 }
2007 `)
2008
2009 expectedVariants := []struct {
2010 moduleName string
2011 variantName string
2012 apkPath string
2013 overrides []string
2014 targetVariant string
2015 packageFlag string
2016 targetPackageFlag string
2017 }{
2018 {
2019 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002020 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002021 overrides: nil,
2022 targetVariant: "android_common",
2023 packageFlag: "",
2024 targetPackageFlag: "",
2025 },
2026 {
2027 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002028 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002029 overrides: []string{"foo_test"},
2030 targetVariant: "android_common_bar",
2031 packageFlag: "com.android.bar.test",
2032 targetPackageFlag: "com.android.bar",
2033 },
2034 }
2035 for _, expected := range expectedVariants {
2036 variant := ctx.ModuleForTests("foo_test", expected.variantName)
2037
2038 // Check the final apk name
2039 outputs := variant.AllOutputs()
2040 expectedApkPath := buildDir + expected.apkPath
2041 found := false
2042 for _, o := range outputs {
2043 if o == expectedApkPath {
2044 found = true
2045 break
2046 }
2047 }
2048 if !found {
2049 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
2050 }
2051
2052 // Check if the overrides field values are correctly aggregated.
2053 mod := variant.Module().(*AndroidTest)
2054 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
2055 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
2056 expected.overrides, mod.appProperties.Overrides)
2057 }
2058
2059 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
2060 javac := variant.Rule("javac")
2061 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
2062 if !strings.Contains(javac.Args["classpath"], turbine) {
2063 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
2064 }
2065
2066 // Check aapt2 flags.
2067 res := variant.Output("package-res.apk")
2068 aapt2Flags := res.Args["flags"]
2069 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammer9f9fd022020-06-18 19:44:06 +00002070 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expected.packageFlag)
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002071 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
2072 }
2073}
2074
Jaewoong Jung39982342020-01-14 10:27:18 -08002075func TestAndroidTest_FixTestConfig(t *testing.T) {
2076 ctx, _ := testJava(t, `
2077 android_app {
2078 name: "foo",
2079 srcs: ["a.java"],
2080 package_name: "com.android.foo",
2081 sdk_version: "current",
2082 }
2083
2084 android_test {
2085 name: "foo_test",
2086 srcs: ["b.java"],
2087 instrumentation_for: "foo",
2088 }
2089
2090 android_test {
2091 name: "bar_test",
2092 srcs: ["b.java"],
2093 package_name: "com.android.bar.test",
2094 instrumentation_for: "foo",
2095 }
2096
2097 override_android_test {
2098 name: "baz_test",
2099 base: "foo_test",
2100 package_name: "com.android.baz.test",
2101 }
2102 `)
2103
2104 testCases := []struct {
2105 moduleName string
2106 variantName string
2107 expectedFlags []string
2108 }{
2109 {
2110 moduleName: "foo_test",
2111 variantName: "android_common",
2112 },
2113 {
2114 moduleName: "bar_test",
2115 variantName: "android_common",
2116 expectedFlags: []string{
2117 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
2118 "--package-name com.android.bar.test",
2119 },
2120 },
2121 {
2122 moduleName: "foo_test",
2123 variantName: "android_common_baz_test",
2124 expectedFlags: []string{
2125 "--manifest " + buildDir +
2126 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
2127 "--package-name com.android.baz.test",
2128 "--test-file-name baz_test.apk",
2129 },
2130 },
2131 }
2132
2133 for _, test := range testCases {
2134 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
2135 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
2136
2137 if len(test.expectedFlags) > 0 {
2138 if params.Rule == nil {
2139 t.Errorf("test_config_fixer was expected to run, but didn't")
2140 } else {
2141 for _, flag := range test.expectedFlags {
2142 if !strings.Contains(params.RuleParams.Command, flag) {
2143 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2144 }
2145 }
2146 }
2147 } else {
2148 if params.Rule != nil {
2149 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2150 }
2151 }
2152
2153 }
2154}
2155
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002156func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002157 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002158 android_app_import {
2159 name: "foo",
2160 apk: "prebuilts/apk/app.apk",
2161 certificate: "platform",
2162 dex_preopt: {
2163 enabled: true,
2164 },
2165 }
2166 `)
2167
2168 variant := ctx.ModuleForTests("foo", "android_common")
2169
2170 // Check dexpreopt outputs.
2171 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2172 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2173 t.Errorf("can't find dexpreopt outputs")
2174 }
2175
2176 // Check cert signing flag.
2177 signedApk := variant.Output("signed/foo.apk")
2178 signingFlag := signedApk.Args["certificates"]
2179 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2180 if expected != signingFlag {
2181 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2182 }
2183}
2184
2185func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002186 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002187 android_app_import {
2188 name: "foo",
2189 apk: "prebuilts/apk/app.apk",
2190 certificate: "platform",
2191 dex_preopt: {
2192 enabled: false,
2193 },
2194 }
2195 `)
2196
2197 variant := ctx.ModuleForTests("foo", "android_common")
2198
2199 // Check dexpreopt outputs. They shouldn't exist.
2200 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2201 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2202 t.Errorf("dexpreopt shouldn't have run.")
2203 }
2204}
2205
2206func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002207 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002208 android_app_import {
2209 name: "foo",
2210 apk: "prebuilts/apk/app.apk",
2211 presigned: true,
2212 dex_preopt: {
2213 enabled: true,
2214 },
2215 }
2216 `)
2217
2218 variant := ctx.ModuleForTests("foo", "android_common")
2219
2220 // Check dexpreopt outputs.
2221 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2222 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2223 t.Errorf("can't find dexpreopt outputs")
2224 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002225 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002226 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2227 t.Errorf("signing rule shouldn't be included.")
2228 }
2229 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2230 t.Errorf("can't find aligning rule")
2231 }
2232}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002233
Liz Kammer2bc57f62020-05-13 15:49:21 -07002234func TestAndroidAppImport_SigningLineage(t *testing.T) {
2235 ctx, _ := testJava(t, `
2236 android_app_import {
2237 name: "foo",
2238 apk: "prebuilts/apk/app.apk",
2239 certificate: "platform",
2240 lineage: "lineage.bin",
2241 }
2242 `)
2243
2244 variant := ctx.ModuleForTests("foo", "android_common")
2245
2246 // Check cert signing lineage flag.
2247 signedApk := variant.Output("signed/foo.apk")
2248 signingFlag := signedApk.Args["flags"]
2249 expected := "--lineage lineage.bin"
2250 if expected != signingFlag {
2251 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2252 }
2253}
2254
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002255func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2256 ctx, _ := testJava(t, `
2257 android_app_import {
2258 name: "foo",
2259 apk: "prebuilts/apk/app.apk",
2260 default_dev_cert: true,
2261 dex_preopt: {
2262 enabled: true,
2263 },
2264 }
2265 `)
2266
2267 variant := ctx.ModuleForTests("foo", "android_common")
2268
2269 // Check dexpreopt outputs.
2270 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2271 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2272 t.Errorf("can't find dexpreopt outputs")
2273 }
2274
2275 // Check cert signing flag.
2276 signedApk := variant.Output("signed/foo.apk")
2277 signingFlag := signedApk.Args["certificates"]
2278 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2279 if expected != signingFlag {
2280 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2281 }
2282}
2283
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002284func TestAndroidAppImport_DpiVariants(t *testing.T) {
2285 bp := `
2286 android_app_import {
2287 name: "foo",
2288 apk: "prebuilts/apk/app.apk",
2289 dpi_variants: {
2290 xhdpi: {
2291 apk: "prebuilts/apk/app_xhdpi.apk",
2292 },
2293 xxhdpi: {
2294 apk: "prebuilts/apk/app_xxhdpi.apk",
2295 },
2296 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002297 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002298 dex_preopt: {
2299 enabled: true,
2300 },
2301 }
2302 `
2303 testCases := []struct {
2304 name string
2305 aaptPreferredConfig *string
2306 aaptPrebuiltDPI []string
2307 expected string
2308 }{
2309 {
2310 name: "no preferred",
2311 aaptPreferredConfig: nil,
2312 aaptPrebuiltDPI: []string{},
2313 expected: "prebuilts/apk/app.apk",
2314 },
2315 {
2316 name: "AAPTPreferredConfig matches",
2317 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002318 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002319 expected: "prebuilts/apk/app_xhdpi.apk",
2320 },
2321 {
2322 name: "AAPTPrebuiltDPI matches",
2323 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2324 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2325 expected: "prebuilts/apk/app_xxhdpi.apk",
2326 },
2327 {
2328 name: "non-first AAPTPrebuiltDPI matches",
2329 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2330 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2331 expected: "prebuilts/apk/app_xhdpi.apk",
2332 },
2333 {
2334 name: "no matches",
2335 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2336 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2337 expected: "prebuilts/apk/app.apk",
2338 },
2339 }
2340
2341 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2342 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002343 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002344 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2345 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002346 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002347
2348 run(t, ctx, config)
2349
2350 variant := ctx.ModuleForTests("foo", "android_common")
2351 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2352 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2353 if len(matches) != 2 {
2354 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2355 }
2356 if test.expected != matches[1] {
2357 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2358 }
2359 }
2360}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002361
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002362func TestAndroidAppImport_Filename(t *testing.T) {
2363 ctx, config := testJava(t, `
2364 android_app_import {
2365 name: "foo",
2366 apk: "prebuilts/apk/app.apk",
2367 presigned: true,
2368 }
2369
2370 android_app_import {
2371 name: "bar",
2372 apk: "prebuilts/apk/app.apk",
2373 presigned: true,
2374 filename: "bar_sample.apk"
2375 }
2376 `)
2377
2378 testCases := []struct {
2379 name string
2380 expected string
2381 }{
2382 {
2383 name: "foo",
2384 expected: "foo.apk",
2385 },
2386 {
2387 name: "bar",
2388 expected: "bar_sample.apk",
2389 },
2390 }
2391
2392 for _, test := range testCases {
2393 variant := ctx.ModuleForTests(test.name, "android_common")
2394 if variant.MaybeOutput(test.expected).Rule == nil {
2395 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2396 }
2397
2398 a := variant.Module().(*AndroidAppImport)
2399 expectedValues := []string{test.expected}
2400 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002401 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002402 if !reflect.DeepEqual(actualValues, expectedValues) {
2403 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2404 actualValues, expectedValues)
2405 }
2406 }
2407}
2408
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002409func TestAndroidAppImport_ArchVariants(t *testing.T) {
2410 // The test config's target arch is ARM64.
2411 testCases := []struct {
2412 name string
2413 bp string
2414 expected string
2415 }{
2416 {
2417 name: "matching arch",
2418 bp: `
2419 android_app_import {
2420 name: "foo",
2421 apk: "prebuilts/apk/app.apk",
2422 arch: {
2423 arm64: {
2424 apk: "prebuilts/apk/app_arm64.apk",
2425 },
2426 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002427 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002428 dex_preopt: {
2429 enabled: true,
2430 },
2431 }
2432 `,
2433 expected: "prebuilts/apk/app_arm64.apk",
2434 },
2435 {
2436 name: "no matching arch",
2437 bp: `
2438 android_app_import {
2439 name: "foo",
2440 apk: "prebuilts/apk/app.apk",
2441 arch: {
2442 arm: {
2443 apk: "prebuilts/apk/app_arm.apk",
2444 },
2445 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002446 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002447 dex_preopt: {
2448 enabled: true,
2449 },
2450 }
2451 `,
2452 expected: "prebuilts/apk/app.apk",
2453 },
2454 }
2455
2456 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2457 for _, test := range testCases {
2458 ctx, _ := testJava(t, test.bp)
2459
2460 variant := ctx.ModuleForTests("foo", "android_common")
2461 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2462 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2463 if len(matches) != 2 {
2464 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2465 }
2466 if test.expected != matches[1] {
2467 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2468 }
2469 }
2470}
2471
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002472func TestAndroidTestImport(t *testing.T) {
2473 ctx, config := testJava(t, `
2474 android_test_import {
2475 name: "foo",
2476 apk: "prebuilts/apk/app.apk",
2477 presigned: true,
2478 data: [
2479 "testdata/data",
2480 ],
2481 }
2482 `)
2483
2484 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2485
2486 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002487 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002488 expected := []string{"tests"}
2489 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2490 if !reflect.DeepEqual(expected, actual) {
2491 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2492 }
2493 expected = []string{"testdata/data:testdata/data"}
2494 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2495 if !reflect.DeepEqual(expected, actual) {
2496 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2497 }
2498}
2499
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002500func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2501 ctx, _ := testJava(t, `
2502 android_test_import {
2503 name: "foo",
2504 apk: "prebuilts/apk/app.apk",
2505 certificate: "cert/new_cert",
2506 data: [
2507 "testdata/data",
2508 ],
2509 }
2510
2511 android_test_import {
2512 name: "foo_presigned",
2513 apk: "prebuilts/apk/app.apk",
2514 presigned: true,
2515 data: [
2516 "testdata/data",
2517 ],
2518 }
2519 `)
2520
2521 variant := ctx.ModuleForTests("foo", "android_common")
2522 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2523 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2524 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2525 }
2526
2527 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2528 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2529 if jniRule != android.Cp.String() {
2530 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2531 }
Liz Kammer7e20dda2020-05-20 14:36:30 -07002532 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2533 t.Errorf("Presigned test apk should be aligned")
2534 }
2535}
2536
2537func TestAndroidTestImport_Preprocessed(t *testing.T) {
2538 ctx, _ := testJava(t, `
2539 android_test_import {
2540 name: "foo",
2541 apk: "prebuilts/apk/app.apk",
2542 presigned: true,
2543 preprocessed: true,
2544 }
2545
2546 android_test_import {
2547 name: "foo_cert",
2548 apk: "prebuilts/apk/app.apk",
2549 certificate: "cert/new_cert",
2550 preprocessed: true,
2551 }
2552 `)
2553
2554 testModules := []string{"foo", "foo_cert"}
2555 for _, m := range testModules {
2556 apkName := m + ".apk"
2557 variant := ctx.ModuleForTests(m, "android_common")
2558 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2559 if jniRule != android.Cp.String() {
2560 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2561 }
2562
2563 // Make sure signing and aligning were skipped.
2564 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2565 t.Errorf("signing rule shouldn't be included for preprocessed.")
2566 }
2567 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2568 t.Errorf("aligning rule shouldn't be for preprocessed")
2569 }
2570 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002571}
2572
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002573func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002574 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002575 cc_library {
2576 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002577 sdk_version: "current",
2578 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002579 }
2580
2581 android_test {
2582 name: "stl",
2583 jni_libs: ["libjni"],
2584 compile_multilib: "both",
2585 sdk_version: "current",
2586 stl: "c++_shared",
2587 }
2588
2589 android_test {
2590 name: "system",
2591 jni_libs: ["libjni"],
2592 compile_multilib: "both",
2593 sdk_version: "current",
2594 }
2595 `)
2596
2597 testCases := []struct {
2598 name string
2599 jnis []string
2600 }{
2601 {"stl",
2602 []string{
2603 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002604 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002605 },
2606 },
2607 {"system",
2608 []string{
2609 "libjni.so",
2610 },
2611 },
2612 }
2613
2614 for _, test := range testCases {
2615 t.Run(test.name, func(t *testing.T) {
2616 app := ctx.ModuleForTests(test.name, "android_common")
2617 jniLibZip := app.Output("jnilibs.zip")
2618 var jnis []string
2619 args := strings.Fields(jniLibZip.Args["jarArgs"])
2620 for i := 0; i < len(args); i++ {
2621 if args[i] == "-f" {
2622 jnis = append(jnis, args[i+1])
2623 i += 1
2624 }
2625 }
2626 jnisJoined := strings.Join(jnis, " ")
2627 for _, jni := range test.jnis {
2628 if !strings.Contains(jnisJoined, jni) {
2629 t.Errorf("missing jni %q in %q", jni, jnis)
2630 }
2631 }
2632 })
2633 }
2634}
Colin Cross50ddcc42019-05-16 12:28:22 -07002635
2636func TestUsesLibraries(t *testing.T) {
2637 bp := `
2638 java_sdk_library {
2639 name: "foo",
2640 srcs: ["a.java"],
2641 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002642 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002643 }
2644
2645 java_sdk_library {
Paul Duffin64e61992020-05-15 10:20:31 +01002646 name: "qux",
2647 srcs: ["a.java"],
2648 api_packages: ["qux"],
2649 sdk_version: "current",
2650 }
2651
2652 java_sdk_library {
2653 name: "quuz",
2654 srcs: ["a.java"],
2655 api_packages: ["quuz"],
2656 sdk_version: "current",
2657 }
2658
2659 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002660 name: "bar",
2661 srcs: ["a.java"],
2662 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002663 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002664 }
2665
2666 android_app {
2667 name: "app",
2668 srcs: ["a.java"],
Paul Duffin64e61992020-05-15 10:20:31 +01002669 libs: ["qux", "quuz.stubs"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002670 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002671 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002672 optional_uses_libs: [
2673 "bar",
2674 "baz",
2675 ],
2676 }
2677
2678 android_app_import {
2679 name: "prebuilt",
2680 apk: "prebuilts/apk/app.apk",
2681 certificate: "platform",
2682 uses_libs: ["foo"],
2683 optional_uses_libs: [
2684 "bar",
2685 "baz",
2686 ],
2687 }
2688 `
2689
Colin Cross98be1bb2019-12-13 20:41:13 -08002690 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002691 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2692
Colin Cross98be1bb2019-12-13 20:41:13 -08002693 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002694
2695 run(t, ctx, config)
2696
2697 app := ctx.ModuleForTests("app", "android_common")
2698 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2699
Paul Duffin64e61992020-05-15 10:20:31 +01002700 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2701 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2702 if w := "--uses-library qux"; !strings.Contains(manifestFixerArgs, w) {
2703 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2704 }
2705 if w := "--uses-library quuz"; !strings.Contains(manifestFixerArgs, w) {
2706 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2707 }
2708
Colin Cross50ddcc42019-05-16 12:28:22 -07002709 // Test that all libraries are verified
2710 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2711 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2712 t.Errorf("wanted %q in %q", w, cmd)
2713 }
2714
2715 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2716 t.Errorf("wanted %q in %q", w, cmd)
2717 }
2718
2719 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2720
2721 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2722 t.Errorf("wanted %q in %q", w, cmd)
2723 }
2724
2725 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2726 t.Errorf("wanted %q in %q", w, cmd)
2727 }
2728
2729 // Test that only present libraries are preopted
2730 cmd = app.Rule("dexpreopt").RuleParams.Command
2731
Ulya Trafimovich5f364b62020-06-30 12:39:01 +01002732 if w := `--target-classpath-for-sdk any /system/framework/foo.jar:/system/framework/bar.jar`; !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002733 t.Errorf("wanted %q in %q", w, cmd)
2734 }
2735
2736 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2737
Ulya Trafimovich5f364b62020-06-30 12:39:01 +01002738 if w := `--target-classpath-for-sdk any /system/framework/foo.jar:/system/framework/bar.jar`; !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002739 t.Errorf("wanted %q in %q", w, cmd)
2740 }
2741}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002742
2743func TestCodelessApp(t *testing.T) {
2744 testCases := []struct {
2745 name string
2746 bp string
2747 noCode bool
2748 }{
2749 {
2750 name: "normal",
2751 bp: `
2752 android_app {
2753 name: "foo",
2754 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002755 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002756 }
2757 `,
2758 noCode: false,
2759 },
2760 {
2761 name: "app without sources",
2762 bp: `
2763 android_app {
2764 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002765 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002766 }
2767 `,
2768 noCode: true,
2769 },
2770 {
2771 name: "app with libraries",
2772 bp: `
2773 android_app {
2774 name: "foo",
2775 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002776 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002777 }
2778
2779 java_library {
2780 name: "lib",
2781 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002782 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002783 }
2784 `,
2785 noCode: false,
2786 },
2787 {
2788 name: "app with sourceless libraries",
2789 bp: `
2790 android_app {
2791 name: "foo",
2792 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002793 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002794 }
2795
2796 java_library {
2797 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002798 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002799 }
2800 `,
2801 // TODO(jungjw): this should probably be true
2802 noCode: false,
2803 },
2804 }
2805
2806 for _, test := range testCases {
2807 t.Run(test.name, func(t *testing.T) {
2808 ctx := testApp(t, test.bp)
2809
2810 foo := ctx.ModuleForTests("foo", "android_common")
2811 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2812 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2813 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2814 }
2815 })
2816 }
2817}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002818
2819func TestEmbedNotice(t *testing.T) {
Colin Cross238c1f32020-06-07 16:58:18 -07002820 ctx, _ := testJavaWithFS(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002821 android_app {
2822 name: "foo",
2823 srcs: ["a.java"],
2824 static_libs: ["javalib"],
2825 jni_libs: ["libjni"],
2826 notice: "APP_NOTICE",
2827 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002828 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002829 }
2830
2831 // No embed_notice flag
2832 android_app {
2833 name: "bar",
2834 srcs: ["a.java"],
2835 jni_libs: ["libjni"],
2836 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002837 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002838 }
2839
2840 // No NOTICE files
2841 android_app {
2842 name: "baz",
2843 srcs: ["a.java"],
2844 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002845 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002846 }
2847
2848 cc_library {
2849 name: "libjni",
2850 system_shared_libs: [],
2851 stl: "none",
2852 notice: "LIB_NOTICE",
Colin Cross1c93c292020-02-15 10:38:00 -08002853 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002854 }
2855
2856 java_library {
2857 name: "javalib",
2858 srcs: [
2859 ":gen",
2860 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002861 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002862 }
2863
2864 genrule {
2865 name: "gen",
2866 tools: ["gentool"],
2867 out: ["gen.java"],
2868 notice: "GENRULE_NOTICE",
2869 }
2870
2871 java_binary_host {
2872 name: "gentool",
2873 srcs: ["b.java"],
2874 notice: "TOOL_NOTICE",
2875 }
Colin Cross238c1f32020-06-07 16:58:18 -07002876 `, map[string][]byte{
2877 "APP_NOTICE": nil,
2878 "GENRULE_NOTICE": nil,
2879 "LIB_NOTICE": nil,
2880 "TOOL_NOTICE": nil,
2881 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002882
2883 // foo has NOTICE files to process, and embed_notices is true.
2884 foo := ctx.ModuleForTests("foo", "android_common")
2885 // verify merge notices rule.
2886 mergeNotices := foo.Rule("mergeNoticesRule")
2887 noticeInputs := mergeNotices.Inputs.Strings()
2888 // TOOL_NOTICE should be excluded as it's a host module.
2889 if len(mergeNotices.Inputs) != 3 {
2890 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2891 }
2892 if !inList("APP_NOTICE", noticeInputs) {
2893 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2894 }
2895 if !inList("LIB_NOTICE", noticeInputs) {
2896 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2897 }
2898 if !inList("GENRULE_NOTICE", noticeInputs) {
2899 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2900 }
2901 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2902 res := foo.Output("package-res.apk")
2903 aapt2Flags := res.Args["flags"]
2904 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2905 if !strings.Contains(aapt2Flags, e) {
2906 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2907 }
2908
2909 // bar has NOTICE files to process, but embed_notices is not set.
2910 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002911 res = bar.Output("package-res.apk")
2912 aapt2Flags = res.Args["flags"]
2913 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2914 if strings.Contains(aapt2Flags, e) {
2915 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002916 }
2917
2918 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2919 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002920 res = baz.Output("package-res.apk")
2921 aapt2Flags = res.Args["flags"]
2922 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2923 if strings.Contains(aapt2Flags, e) {
2924 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002925 }
2926}
Colin Cross53a87f52019-06-25 13:35:30 -07002927
2928func TestUncompressDex(t *testing.T) {
2929 testCases := []struct {
2930 name string
2931 bp string
2932
2933 uncompressedPlatform bool
2934 uncompressedUnbundled bool
2935 }{
2936 {
2937 name: "normal",
2938 bp: `
2939 android_app {
2940 name: "foo",
2941 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002942 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002943 }
2944 `,
2945 uncompressedPlatform: true,
2946 uncompressedUnbundled: false,
2947 },
2948 {
2949 name: "use_embedded_dex",
2950 bp: `
2951 android_app {
2952 name: "foo",
2953 use_embedded_dex: true,
2954 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002955 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002956 }
2957 `,
2958 uncompressedPlatform: true,
2959 uncompressedUnbundled: true,
2960 },
2961 {
2962 name: "privileged",
2963 bp: `
2964 android_app {
2965 name: "foo",
2966 privileged: true,
2967 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002968 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002969 }
2970 `,
2971 uncompressedPlatform: true,
2972 uncompressedUnbundled: true,
2973 },
David Srbecky98c71222020-05-20 22:20:28 +01002974 {
2975 name: "normal_uncompress_dex_true",
2976 bp: `
2977 android_app {
2978 name: "foo",
2979 srcs: ["a.java"],
2980 sdk_version: "current",
2981 uncompress_dex: true,
2982 }
2983 `,
2984 uncompressedPlatform: true,
2985 uncompressedUnbundled: true,
2986 },
2987 {
2988 name: "normal_uncompress_dex_false",
2989 bp: `
2990 android_app {
2991 name: "foo",
2992 srcs: ["a.java"],
2993 sdk_version: "current",
2994 uncompress_dex: false,
2995 }
2996 `,
2997 uncompressedPlatform: false,
2998 uncompressedUnbundled: false,
2999 },
Colin Cross53a87f52019-06-25 13:35:30 -07003000 }
3001
3002 test := func(t *testing.T, bp string, want bool, unbundled bool) {
3003 t.Helper()
3004
Colin Cross98be1bb2019-12-13 20:41:13 -08003005 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07003006 if unbundled {
3007 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
3008 }
3009
Colin Cross98be1bb2019-12-13 20:41:13 -08003010 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07003011
3012 run(t, ctx, config)
3013
3014 foo := ctx.ModuleForTests("foo", "android_common")
3015 dex := foo.Rule("r8")
3016 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
3017 aligned := foo.MaybeRule("zipalign").Rule != nil
3018
3019 if uncompressedInDexJar != want {
3020 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
3021 }
3022
3023 if aligned != want {
3024 t.Errorf("want aligned %v, got %v", want, aligned)
3025 }
3026 }
3027
3028 for _, tt := range testCases {
3029 t.Run(tt.name, func(t *testing.T) {
3030 t.Run("platform", func(t *testing.T) {
3031 test(t, tt.bp, tt.uncompressedPlatform, false)
3032 })
3033 t.Run("unbundled", func(t *testing.T) {
3034 test(t, tt.bp, tt.uncompressedUnbundled, true)
3035 })
3036 })
3037 }
3038}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07003039
3040func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
3041 if expectedValue != "" {
3042 expectedFlag := "--" + flagName + " " + expectedValue
3043 if !strings.Contains(aapt2Flags, expectedFlag) {
3044 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
3045 }
3046 } else {
3047 unexpectedFlag := "--" + flagName
3048 if strings.Contains(aapt2Flags, unexpectedFlag) {
3049 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
3050 }
3051 }
3052}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003053
3054func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungca095d72020-04-09 16:15:30 -07003055 fs := map[string][]byte{
3056 "baz/res/res/values/strings.xml": nil,
3057 "bar/res/res/values/strings.xml": nil,
3058 }
3059 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003060 runtime_resource_overlay {
3061 name: "foo",
3062 certificate: "platform",
Liz Kammer7fe241f2020-05-19 16:15:25 -07003063 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003064 product_specific: true,
Jaewoong Jungca095d72020-04-09 16:15:30 -07003065 static_libs: ["bar"],
3066 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003067 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003068 }
3069
3070 runtime_resource_overlay {
3071 name: "foo_themed",
3072 certificate: "platform",
3073 product_specific: true,
3074 theme: "faza",
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003075 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003076 }
Jaewoong Jungca095d72020-04-09 16:15:30 -07003077
3078 android_library {
3079 name: "bar",
3080 resource_dirs: ["bar/res"],
3081 }
3082
3083 android_app {
3084 name: "baz",
3085 sdk_version: "current",
3086 resource_dirs: ["baz/res"],
3087 }
3088 `
3089 config := testAppConfig(nil, bp, fs)
3090 ctx := testContext()
3091 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003092
3093 m := ctx.ModuleForTests("foo", "android_common")
3094
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003095 // Check AAPT2 link flags.
3096 aapt2Flags := m.Output("package-res.apk").Args["flags"]
3097 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3098 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
3099 if len(absentFlags) > 0 {
3100 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3101 }
3102
Jaewoong Jungca095d72020-04-09 16:15:30 -07003103 // Check overlay.list output for static_libs dependency.
3104 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
3105 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
3106 if !inList(staticLibPackage, overlayList) {
3107 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
3108 }
3109
3110 // Check AAPT2 link flags for resource_libs dependency.
3111 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
3112 if !strings.Contains(aapt2Flags, resourceLibFlag) {
3113 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
3114 }
3115
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003116 // Check cert signing flag.
3117 signedApk := m.Output("signed/foo.apk")
Liz Kammer7fe241f2020-05-19 16:15:25 -07003118 lineageFlag := signedApk.Args["flags"]
3119 expectedLineageFlag := "--lineage lineage.bin"
3120 if expectedLineageFlag != lineageFlag {
3121 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
3122 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003123 signingFlag := signedApk.Args["certificates"]
3124 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
3125 if expected != signingFlag {
3126 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
3127 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003128 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3129 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003130 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
3131 if !reflect.DeepEqual(path, expectedPath) {
3132 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
3133 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003134
3135 // Check device location.
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003136 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003137 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003138 if !reflect.DeepEqual(path, expectedPath) {
3139 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3140 }
3141
3142 // A themed module has a different device location
3143 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003144 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3145 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003146 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3147 if !reflect.DeepEqual(path, expectedPath) {
3148 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3149 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07003150
3151 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3152 expectedOverrides := []string{"foo"}
3153 if !reflect.DeepEqual(overrides, expectedOverrides) {
3154 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3155 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003156}
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07003157
3158func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3159 ctx, config := testJava(t, `
3160 java_defaults {
3161 name: "rro_defaults",
3162 theme: "default_theme",
3163 product_specific: true,
3164 aaptflags: ["--keep-raw-values"],
3165 }
3166
3167 runtime_resource_overlay {
3168 name: "foo_with_defaults",
3169 defaults: ["rro_defaults"],
3170 }
3171
3172 runtime_resource_overlay {
3173 name: "foo_barebones",
3174 }
3175 `)
3176
3177 //
3178 // RRO module with defaults
3179 //
3180 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3181
3182 // Check AAPT2 link flags.
3183 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3184 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3185 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3186 if len(absentFlags) > 0 {
3187 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3188 }
3189
3190 // Check device location.
3191 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3192 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3193 if !reflect.DeepEqual(path, expectedPath) {
3194 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3195 }
3196
3197 //
3198 // RRO module without defaults
3199 //
3200 m = ctx.ModuleForTests("foo_barebones", "android_common")
3201
3202 // Check AAPT2 link flags.
3203 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3204 unexpectedFlags := "--keep-raw-values"
3205 if inList(unexpectedFlags, aapt2Flags) {
3206 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3207 }
3208
3209 // Check device location.
3210 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3211 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3212 if !reflect.DeepEqual(path, expectedPath) {
3213 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3214 }
3215}
Roshan Piusb8307962020-04-27 09:42:27 -07003216
3217func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3218 ctx, _ := testJava(t, `
3219 runtime_resource_overlay {
3220 name: "foo_overlay",
3221 certificate: "platform",
3222 product_specific: true,
3223 sdk_version: "current",
3224 }
3225
3226 override_runtime_resource_overlay {
3227 name: "bar_overlay",
3228 base: "foo_overlay",
3229 package_name: "com.android.bar.overlay",
3230 target_package_name: "com.android.bar",
3231 }
3232 `)
3233
3234 expectedVariants := []struct {
3235 moduleName string
3236 variantName string
3237 apkPath string
3238 overrides []string
3239 targetVariant string
3240 packageFlag string
3241 targetPackageFlag string
3242 }{
3243 {
3244 variantName: "android_common",
3245 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3246 overrides: nil,
3247 targetVariant: "android_common",
3248 packageFlag: "",
3249 targetPackageFlag: "",
3250 },
3251 {
3252 variantName: "android_common_bar_overlay",
3253 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3254 overrides: []string{"foo_overlay"},
3255 targetVariant: "android_common_bar",
3256 packageFlag: "com.android.bar.overlay",
3257 targetPackageFlag: "com.android.bar",
3258 },
3259 }
3260 for _, expected := range expectedVariants {
3261 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3262
3263 // Check the final apk name
3264 outputs := variant.AllOutputs()
3265 expectedApkPath := buildDir + expected.apkPath
3266 found := false
3267 for _, o := range outputs {
3268 if o == expectedApkPath {
3269 found = true
3270 break
3271 }
3272 }
3273 if !found {
3274 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3275 }
3276
3277 // Check if the overrides field values are correctly aggregated.
3278 mod := variant.Module().(*RuntimeResourceOverlay)
3279 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3280 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3281 expected.overrides, mod.properties.Overrides)
3282 }
3283
3284 // Check aapt2 flags.
3285 res := variant.Output("package-res.apk")
3286 aapt2Flags := res.Args["flags"]
3287 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammer9f9fd022020-06-18 19:44:06 +00003288 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", "")
Roshan Piusb8307962020-04-27 09:42:27 -07003289 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3290 }
3291}