blob: 4347db8b1b21afa67543f630ece03d99f15bac24 [file] [log] [blame]
Colin Cross3bc7ffa2017-11-22 16:19:37 -08001// Copyright 2017 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package java
16
17import (
Colin Crossd09b0b62018-04-18 11:06:47 -070018 "fmt"
Colin Crossa4f08812018-10-02 22:03:40 -070019 "path/filepath"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080020 "reflect"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070021 "regexp"
Colin Crossb69301e2017-12-01 10:48:26 -080022 "sort"
Colin Crossd09b0b62018-04-18 11:06:47 -070023 "strings"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080024 "testing"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070025
26 "github.com/google/blueprint/proptools"
27
28 "android/soong/android"
29 "android/soong/cc"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080030)
31
32var (
33 resourceFiles = []string{
34 "res/layout/layout.xml",
35 "res/values/strings.xml",
36 "res/values-en-rUS/strings.xml",
37 }
38
39 compiledResourceFiles = []string{
40 "aapt2/res/layout_layout.xml.flat",
41 "aapt2/res/values_strings.arsc.flat",
42 "aapt2/res/values-en-rUS_strings.arsc.flat",
43 }
44)
45
Colin Cross98be1bb2019-12-13 20:41:13 -080046func testAppConfig(env map[string]string, bp string, fs map[string][]byte) android.Config {
Colin Cross527012a2017-11-30 22:56:16 -080047 appFS := map[string][]byte{}
48 for k, v := range fs {
49 appFS[k] = v
Colin Cross3bc7ffa2017-11-22 16:19:37 -080050 }
51
Colin Cross527012a2017-11-30 22:56:16 -080052 for _, file := range resourceFiles {
53 appFS[file] = nil
54 }
55
Colin Cross98be1bb2019-12-13 20:41:13 -080056 return testConfig(env, bp, appFS)
Colin Cross527012a2017-11-30 22:56:16 -080057}
58
59func testApp(t *testing.T, bp string) *android.TestContext {
Colin Cross98be1bb2019-12-13 20:41:13 -080060 config := testAppConfig(nil, bp, nil)
Colin Cross527012a2017-11-30 22:56:16 -080061
Colin Cross98be1bb2019-12-13 20:41:13 -080062 ctx := testContext()
Colin Cross527012a2017-11-30 22:56:16 -080063
64 run(t, ctx, config)
65
66 return ctx
Colin Cross3bc7ffa2017-11-22 16:19:37 -080067}
68
69func TestApp(t *testing.T) {
Colin Crossa97c5d32018-03-28 14:58:31 -070070 for _, moduleType := range []string{"android_app", "android_library"} {
71 t.Run(moduleType, func(t *testing.T) {
72 ctx := testApp(t, moduleType+` {
73 name: "foo",
74 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +090075 sdk_version: "current"
Colin Crossa97c5d32018-03-28 14:58:31 -070076 }
77 `)
Colin Cross3bc7ffa2017-11-22 16:19:37 -080078
Colin Crossa97c5d32018-03-28 14:58:31 -070079 foo := ctx.ModuleForTests("foo", "android_common")
Colin Cross3bc7ffa2017-11-22 16:19:37 -080080
Colin Cross31656952018-05-24 16:11:20 -070081 var expectedLinkImplicits []string
82
83 manifestFixer := foo.Output("manifest_fixer/AndroidManifest.xml")
84 expectedLinkImplicits = append(expectedLinkImplicits, manifestFixer.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080085
Colin Crossa97c5d32018-03-28 14:58:31 -070086 frameworkRes := ctx.ModuleForTests("framework-res", "android_common")
87 expectedLinkImplicits = append(expectedLinkImplicits,
88 frameworkRes.Output("package-res.apk").Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080089
Colin Crossa97c5d32018-03-28 14:58:31 -070090 // Test the mapping from input files to compiled output file names
91 compile := foo.Output(compiledResourceFiles[0])
92 if !reflect.DeepEqual(resourceFiles, compile.Inputs.Strings()) {
93 t.Errorf("expected aapt2 compile inputs expected:\n %#v\n got:\n %#v",
94 resourceFiles, compile.Inputs.Strings())
95 }
Colin Crossb69301e2017-12-01 10:48:26 -080096
Colin Crossa97c5d32018-03-28 14:58:31 -070097 compiledResourceOutputs := compile.Outputs.Strings()
98 sort.Strings(compiledResourceOutputs)
Colin Crossb69301e2017-12-01 10:48:26 -080099
Colin Crossa97c5d32018-03-28 14:58:31 -0700100 expectedLinkImplicits = append(expectedLinkImplicits, compiledResourceOutputs...)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800101
Colin Crossa97c5d32018-03-28 14:58:31 -0700102 list := foo.Output("aapt2/res.list")
103 expectedLinkImplicits = append(expectedLinkImplicits, list.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800104
Colin Crossa97c5d32018-03-28 14:58:31 -0700105 // Check that the link rule uses
106 res := ctx.ModuleForTests("foo", "android_common").Output("package-res.apk")
107 if !reflect.DeepEqual(expectedLinkImplicits, res.Implicits.Strings()) {
108 t.Errorf("expected aapt2 link implicits expected:\n %#v\n got:\n %#v",
109 expectedLinkImplicits, res.Implicits.Strings())
110 }
111 })
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800112 }
113}
Colin Cross890ff552017-11-30 20:13:19 -0800114
Colin Crosse560c4a2019-03-19 16:03:11 -0700115func TestAppSplits(t *testing.T) {
116 ctx := testApp(t, `
117 android_app {
118 name: "foo",
119 srcs: ["a.java"],
120 package_splits: ["v4", "v7,hdpi"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900121 sdk_version: "current"
Colin Crosse560c4a2019-03-19 16:03:11 -0700122 }`)
123
124 foo := ctx.ModuleForTests("foo", "android_common")
125
126 expectedOutputs := []string{
127 filepath.Join(buildDir, ".intermediates/foo/android_common/foo.apk"),
128 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v4.apk"),
129 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v7_hdpi.apk"),
130 }
131 for _, expectedOutput := range expectedOutputs {
132 foo.Output(expectedOutput)
133 }
134
Colin Cross41955e82019-05-29 14:40:35 -0700135 outputFiles, err := foo.Module().(*AndroidApp).OutputFiles("")
136 if err != nil {
137 t.Fatal(err)
138 }
139 if g, w := outputFiles.Strings(), expectedOutputs; !reflect.DeepEqual(g, w) {
140 t.Errorf(`want OutputFiles("") = %q, got %q`, w, g)
Colin Crosse560c4a2019-03-19 16:03:11 -0700141 }
142}
143
Sasha Smundaka7856c02020-04-23 09:49:59 -0700144func TestAndroidAppSet(t *testing.T) {
145 ctx, config := testJava(t, `
146 android_app_set {
147 name: "foo",
148 set: "prebuilts/apks/app.apks",
149 prerelease: true,
Jaewoong Jung11c1e0f2020-06-29 19:18:44 -0700150 }`)
Sasha Smundaka7856c02020-04-23 09:49:59 -0700151 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundak18d98bc2020-05-27 16:36:07 -0700152 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-04-23 09:49:59 -0700153 params := module.Output(packedSplitApks)
154 if params.Rule == nil {
155 t.Errorf("expected output %s is missing", packedSplitApks)
156 }
157 if s := params.Args["allow-prereleased"]; s != "true" {
158 t.Errorf("wrong allow-prereleased value: '%s', expected 'true'", s)
159 }
Jaewoong Jung11c1e0f2020-06-29 19:18:44 -0700160 if s := params.Args["partition"]; s != "system" {
161 t.Errorf("wrong partition value: '%s', expected 'system'", s)
162 }
Sasha Smundaka7856c02020-04-23 09:49:59 -0700163 mkEntries := android.AndroidMkEntriesForTest(t, config, "", module.Module())[0]
Liz Kammercada8072020-07-28 15:47:38 -0700164 actualInstallFile := mkEntries.EntryMap["LOCAL_APK_SET_INSTALL_FILE"]
165 expectedInstallFile := []string{"foo.apk"}
166 if !reflect.DeepEqual(actualInstallFile, expectedInstallFile) {
167 t.Errorf("Unexpected LOCAL_APK_SET_INSTALL_FILE value: '%s', expected: '%s',",
168 actualInstallFile, expectedInstallFile)
Sasha Smundaka7856c02020-04-23 09:49:59 -0700169 }
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 Smundaka7856c02020-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 Smundaka7856c02020-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 Smundaka7856c02020-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 Smundaka7856c02020-04-23 09:49:59 -0700223 ctx := testContext()
224 run(t, ctx, config)
225 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundak18d98bc2020-05-27 16:36:07 -0700226 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-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 Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100372 min_sdk_version: "29",
Artur Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100382 min_sdk_version: "29",
Artur Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100392 min_sdk_version: "29",
Artur Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100402 min_sdk_version: "29",
Artur Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100412 min_sdk_version: "29",
Artur Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100422 min_sdk_version: "29",
Artur Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100432 min_sdk_version: "29",
Artur Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100443 min_sdk_version: "29",
Artur Satayev2db1c3f2020-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 Satayevf40fc852020-04-16 13:43:02 +0100454 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100455 }`,
456 expectedError: "Updatable apps must use stable SDK",
457 },
Artur Satayevf40fc852020-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 Satayev2db1c3f2020-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 Hanbbc3fb72020-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 }
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900536 `
537 fs := map[string][]byte{
538 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
539 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
540 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
541 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
542 }
543
544 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
545
546 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
547 var crtbeginFound, crtendFound bool
Dan Albert92fe7402020-07-15 13:33:30 -0700548 expectedCrtBegin := ctx.ModuleForTests("crtbegin_so",
549 "android_arm64_armv8-a_sdk_29").Rule("partialLd").Output
550 expectedCrtEnd := ctx.ModuleForTests("crtend_so",
551 "android_arm64_armv8-a_sdk_29").Rule("partialLd").Output
552 implicits := []string{}
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900553 for _, input := range inputs {
Dan Albert92fe7402020-07-15 13:33:30 -0700554 implicits = append(implicits, input.String())
555 if strings.HasSuffix(input.String(), expectedCrtBegin.String()) {
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900556 crtbeginFound = true
Dan Albert92fe7402020-07-15 13:33:30 -0700557 } else if strings.HasSuffix(input.String(), expectedCrtEnd.String()) {
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900558 crtendFound = true
559 }
560 }
Dan Albert92fe7402020-07-15 13:33:30 -0700561 if !crtbeginFound {
562 t.Error(fmt.Sprintf(
563 "expected implicit with suffix %q, have the following implicits:\n%s",
564 expectedCrtBegin, strings.Join(implicits, "\n")))
565 }
566 if !crtendFound {
567 t.Error(fmt.Sprintf(
568 "expected implicit with suffix %q, have the following implicits:\n%s",
569 expectedCrtEnd, strings.Join(implicits, "\n")))
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900570 }
571}
572
573func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
574 bp := cc.GatherRequiredDepsForTest(android.Android) + `
575 android_app {
576 name: "foo",
577 srcs: ["a.java"],
578 updatable: true,
579 sdk_version: "current",
580 min_sdk_version: "29", // this APK should support 29
581 jni_libs: ["libjni"],
582 }
583
584 cc_library {
585 name: "libjni",
586 stl: "none",
587 sdk_version: "current",
588 }
589 `
590 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
591}
592
593func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
594 bp := cc.GatherRequiredDepsForTest(android.Android) + `
595 android_app {
596 name: "foo",
597 srcs: ["a.java"],
598 updatable: true,
599 sdk_version: "current",
600 min_sdk_version: "29", // this APK should support 29
601 jni_libs: ["libjni"],
602 }
603
604 cc_library {
605 name: "libjni",
606 stl: "none",
607 shared_libs: ["libbar"],
608 system_shared_libs: [],
609 sdk_version: "27",
610 }
611
612 cc_library {
613 name: "libbar",
614 stl: "none",
615 system_shared_libs: [],
616 sdk_version: "current",
617 }
618 `
619 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
620}
621
Colin Cross0ddae7f2019-02-07 15:30:01 -0800622func TestResourceDirs(t *testing.T) {
623 testCases := []struct {
624 name string
625 prop string
626 resources []string
627 }{
628 {
629 name: "no resource_dirs",
630 prop: "",
631 resources: []string{"res/res/values/strings.xml"},
632 },
633 {
634 name: "resource_dirs",
635 prop: `resource_dirs: ["res"]`,
636 resources: []string{"res/res/values/strings.xml"},
637 },
638 {
639 name: "empty resource_dirs",
640 prop: `resource_dirs: []`,
641 resources: nil,
642 },
643 }
644
645 fs := map[string][]byte{
646 "res/res/values/strings.xml": nil,
647 }
648
649 bp := `
650 android_app {
651 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900652 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800653 %s
654 }
655 `
656
657 for _, testCase := range testCases {
658 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800659 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
660 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800661 run(t, ctx, config)
662
663 module := ctx.ModuleForTests("foo", "android_common")
664 resourceList := module.MaybeOutput("aapt2/res.list")
665
666 var resources []string
667 if resourceList.Rule != nil {
668 for _, compiledResource := range resourceList.Inputs.Strings() {
669 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
670 }
671 }
672
673 if !reflect.DeepEqual(resources, testCase.resources) {
674 t.Errorf("expected resource files %q, got %q",
675 testCase.resources, resources)
676 }
677 })
678 }
679}
680
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800681func TestLibraryAssets(t *testing.T) {
682 bp := `
683 android_app {
684 name: "foo",
685 sdk_version: "current",
686 static_libs: ["lib1", "lib2", "lib3"],
687 }
688
689 android_library {
690 name: "lib1",
691 sdk_version: "current",
692 asset_dirs: ["assets_a"],
693 }
694
695 android_library {
696 name: "lib2",
697 sdk_version: "current",
698 }
699
700 android_library {
701 name: "lib3",
702 sdk_version: "current",
703 static_libs: ["lib4"],
704 }
705
706 android_library {
707 name: "lib4",
708 sdk_version: "current",
709 asset_dirs: ["assets_b"],
710 }
711 `
712
713 testCases := []struct {
714 name string
715 assetFlag string
716 assetPackages []string
717 }{
718 {
719 name: "foo",
720 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
721 assetPackages: []string{
722 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
723 buildDir + "/.intermediates/lib1/android_common/assets.zip",
724 buildDir + "/.intermediates/lib3/android_common/assets.zip",
725 },
726 },
727 {
728 name: "lib1",
729 assetFlag: "-A assets_a",
730 },
731 {
732 name: "lib2",
733 },
734 {
735 name: "lib3",
736 assetPackages: []string{
737 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
738 buildDir + "/.intermediates/lib4/android_common/assets.zip",
739 },
740 },
741 {
742 name: "lib4",
743 assetFlag: "-A assets_b",
744 },
745 }
746 ctx := testApp(t, bp)
747
748 for _, test := range testCases {
749 t.Run(test.name, func(t *testing.T) {
750 m := ctx.ModuleForTests(test.name, "android_common")
751
752 // Check asset flag in aapt2 link flags
753 var aapt2link android.TestingBuildParams
754 if len(test.assetPackages) > 0 {
755 aapt2link = m.Output("aapt2/package-res.apk")
756 } else {
757 aapt2link = m.Output("package-res.apk")
758 }
759 aapt2Flags := aapt2link.Args["flags"]
760 if test.assetFlag != "" {
761 if !strings.Contains(aapt2Flags, test.assetFlag) {
762 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
763 }
764 } else {
765 if strings.Contains(aapt2Flags, " -A ") {
766 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
767 }
768 }
769
770 // Check asset merge rule.
771 if len(test.assetPackages) > 0 {
772 mergeAssets := m.Output("package-res.apk")
773 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
774 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
775 mergeAssets.Inputs.Strings(), test.assetPackages)
776 }
777 }
778 })
779 }
780}
781
Colin Crossbec85302019-02-13 13:15:46 -0800782func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800783 testCases := []struct {
784 name string
785 enforceRROTargets []string
786 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800787 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800788 overlayFiles map[string][]string
789 rroDirs map[string][]string
790 }{
791 {
792 name: "no RRO",
793 enforceRROTargets: nil,
794 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800795 resourceFiles: map[string][]string{
796 "foo": nil,
797 "bar": {"bar/res/res/values/strings.xml"},
798 "lib": nil,
799 "lib2": {"lib2/res/res/values/strings.xml"},
800 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800801 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800802 "foo": {
803 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800804 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000805 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800806 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800807 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
808 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000809 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800810 },
Colin Crossbec85302019-02-13 13:15:46 -0800811 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800812 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
813 "device/vendor/blah/overlay/bar/res/values/strings.xml",
814 },
Colin Crossbec85302019-02-13 13:15:46 -0800815 "lib": {
816 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
817 "lib/res/res/values/strings.xml",
818 "device/vendor/blah/overlay/lib/res/values/strings.xml",
819 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800820 },
821 rroDirs: map[string][]string{
822 "foo": nil,
823 "bar": nil,
824 },
825 },
826 {
827 name: "enforce RRO on foo",
828 enforceRROTargets: []string{"foo"},
829 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800830 resourceFiles: map[string][]string{
831 "foo": nil,
832 "bar": {"bar/res/res/values/strings.xml"},
833 "lib": nil,
834 "lib2": {"lib2/res/res/values/strings.xml"},
835 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800836 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800837 "foo": {
838 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800839 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000840 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800841 "foo/res/res/values/strings.xml",
842 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
843 },
Colin Crossbec85302019-02-13 13:15:46 -0800844 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800845 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
846 "device/vendor/blah/overlay/bar/res/values/strings.xml",
847 },
Colin Crossbec85302019-02-13 13:15:46 -0800848 "lib": {
849 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
850 "lib/res/res/values/strings.xml",
851 "device/vendor/blah/overlay/lib/res/values/strings.xml",
852 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800853 },
Colin Crossc1c37552019-01-31 11:42:41 -0800854
Colin Cross5c4791c2019-02-01 11:44:44 -0800855 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800856 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000857 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800858 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
859 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000860 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800861 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800862 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800863 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800864 },
865 },
866 {
867 name: "enforce RRO on all",
868 enforceRROTargets: []string{"*"},
869 enforceRROExcludedOverlays: []string{
870 // Excluding specific apps/res directories also allowed.
871 "device/vendor/blah/static_overlay/foo",
872 "device/vendor/blah/static_overlay/bar/res",
873 },
Colin Crossbec85302019-02-13 13:15:46 -0800874 resourceFiles: map[string][]string{
875 "foo": nil,
876 "bar": {"bar/res/res/values/strings.xml"},
877 "lib": nil,
878 "lib2": {"lib2/res/res/values/strings.xml"},
879 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800880 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800881 "foo": {
882 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800883 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000884 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800885 "foo/res/res/values/strings.xml",
886 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
887 },
Colin Crossbec85302019-02-13 13:15:46 -0800888 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
889 "lib": {
890 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
891 "lib/res/res/values/strings.xml",
892 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800893 },
894 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800895 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000896 "device:device/vendor/blah/overlay/foo/res",
897 "product:product/vendor/blah/overlay/foo/res",
898 // Lib dep comes after the direct deps
899 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800900 },
Anton Hansson53c88442019-03-18 15:53:16 +0000901 "bar": {"device:device/vendor/blah/overlay/bar/res"},
902 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800903 },
904 },
905 }
906
Anton Hansson53c88442019-03-18 15:53:16 +0000907 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800908 "device/vendor/blah/overlay",
909 "device/vendor/blah/overlay2",
910 "device/vendor/blah/static_overlay",
911 }
912
Anton Hansson53c88442019-03-18 15:53:16 +0000913 productResourceOverlays := []string{
914 "product/vendor/blah/overlay",
915 }
916
Colin Cross890ff552017-11-30 20:13:19 -0800917 fs := map[string][]byte{
918 "foo/res/res/values/strings.xml": nil,
919 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800920 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800921 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800922 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
923 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800924 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800925 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
926 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
927 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000928 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800929 }
930
931 bp := `
932 android_app {
933 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900934 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800935 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000936 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800937 }
938
939 android_app {
940 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900941 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800942 resource_dirs: ["bar/res"],
943 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800944
945 android_library {
946 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900947 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800948 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800949 static_libs: ["lib2"],
950 }
951
952 android_library {
953 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900954 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800955 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800956 }
Anton Hansson53c88442019-03-18 15:53:16 +0000957
958 // This library has the same resources as lib (should not lead to dupe RROs)
959 android_library {
960 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900961 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000962 resource_dirs: ["lib/res"]
963 }
Colin Cross890ff552017-11-30 20:13:19 -0800964 `
965
Colin Cross5c4791c2019-02-01 11:44:44 -0800966 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800967 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800968 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000969 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
970 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800971 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800972 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800973 }
974 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800975 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800976 }
977
Colin Cross98be1bb2019-12-13 20:41:13 -0800978 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800979 run(t, ctx, config)
980
Colin Crossbec85302019-02-13 13:15:46 -0800981 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
982 for _, o := range list {
983 res := module.MaybeOutput(o)
984 if res.Rule != nil {
985 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
986 // verify the inputs to the .arsc.flat rule.
987 files = append(files, res.Inputs.Strings()...)
988 } else {
989 // Otherwise, verify the full path to the output of the other module
990 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000991 }
Colin Cross890ff552017-11-30 20:13:19 -0800992 }
Colin Crossbec85302019-02-13 13:15:46 -0800993 return files
Colin Cross890ff552017-11-30 20:13:19 -0800994 }
995
Colin Crossbec85302019-02-13 13:15:46 -0800996 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
997 module := ctx.ModuleForTests(moduleName, "android_common")
998 resourceList := module.MaybeOutput("aapt2/res.list")
999 if resourceList.Rule != nil {
1000 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +00001001 }
Colin Crossbec85302019-02-13 13:15:46 -08001002 overlayList := module.MaybeOutput("aapt2/overlay.list")
1003 if overlayList.Rule != nil {
1004 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
1005 }
1006
Anton Hansson53c88442019-03-18 15:53:16 +00001007 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
1008 var prefix string
1009 if d.overlayType == device {
1010 prefix = "device:"
1011 } else if d.overlayType == product {
1012 prefix = "product:"
1013 } else {
1014 t.Fatalf("Unexpected overlayType %d", d.overlayType)
1015 }
1016 rroDirs = append(rroDirs, prefix+d.path.String())
1017 }
Colin Crossbec85302019-02-13 13:15:46 -08001018
1019 return resourceFiles, overlayFiles, rroDirs
1020 }
1021
1022 modules := []string{"foo", "bar", "lib", "lib2"}
1023 for _, module := range modules {
1024 resourceFiles, overlayFiles, rroDirs := getResources(module)
1025
1026 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
1027 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
1028 module, testCase.resourceFiles[module], resourceFiles)
1029 }
1030 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
1031 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
1032 module, testCase.overlayFiles[module], overlayFiles)
1033 }
1034 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +00001035 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -08001036 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +00001037 }
Colin Cross890ff552017-11-30 20:13:19 -08001038 }
Colin Cross890ff552017-11-30 20:13:19 -08001039 })
1040 }
1041}
Colin Crossd09b0b62018-04-18 11:06:47 -07001042
Jeongik Cha219141c2020-08-06 23:00:37 +09001043func checkSdkVersion(t *testing.T, config android.Config, expectedSdkVersion string) {
1044 ctx := testContext()
1045
1046 run(t, ctx, config)
1047
1048 foo := ctx.ModuleForTests("foo", "android_common")
1049 link := foo.Output("package-res.apk")
1050 linkFlags := strings.Split(link.Args["flags"], " ")
1051 min := android.IndexList("--min-sdk-version", linkFlags)
1052 target := android.IndexList("--target-sdk-version", linkFlags)
1053
1054 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1055 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1056 }
1057
1058 gotMinSdkVersion := linkFlags[min+1]
1059 gotTargetSdkVersion := linkFlags[target+1]
1060
1061 if gotMinSdkVersion != expectedSdkVersion {
1062 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1063 expectedSdkVersion, gotMinSdkVersion)
1064 }
1065
1066 if gotTargetSdkVersion != expectedSdkVersion {
1067 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1068 expectedSdkVersion, gotTargetSdkVersion)
1069 }
1070}
1071
Colin Crossd09b0b62018-04-18 11:06:47 -07001072func TestAppSdkVersion(t *testing.T) {
1073 testCases := []struct {
1074 name string
1075 sdkVersion string
1076 platformSdkInt int
1077 platformSdkCodename string
1078 platformSdkFinal bool
1079 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +09001080 platformApis bool
Dan Albert4f378d72020-07-23 17:32:15 -07001081 activeCodenames []string
Colin Crossd09b0b62018-04-18 11:06:47 -07001082 }{
1083 {
1084 name: "current final SDK",
1085 sdkVersion: "current",
1086 platformSdkInt: 27,
1087 platformSdkCodename: "REL",
1088 platformSdkFinal: true,
1089 expectedMinSdkVersion: "27",
1090 },
1091 {
1092 name: "current non-final SDK",
1093 sdkVersion: "current",
1094 platformSdkInt: 27,
1095 platformSdkCodename: "OMR1",
1096 platformSdkFinal: false,
1097 expectedMinSdkVersion: "OMR1",
Dan Albert4f378d72020-07-23 17:32:15 -07001098 activeCodenames: []string{"OMR1"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001099 },
1100 {
1101 name: "default final SDK",
1102 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001103 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001104 platformSdkInt: 27,
1105 platformSdkCodename: "REL",
1106 platformSdkFinal: true,
1107 expectedMinSdkVersion: "27",
1108 },
1109 {
1110 name: "default non-final SDK",
1111 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001112 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001113 platformSdkInt: 27,
1114 platformSdkCodename: "OMR1",
1115 platformSdkFinal: false,
1116 expectedMinSdkVersion: "OMR1",
Dan Albert4f378d72020-07-23 17:32:15 -07001117 activeCodenames: []string{"OMR1"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001118 },
1119 {
1120 name: "14",
1121 sdkVersion: "14",
1122 expectedMinSdkVersion: "14",
Dan Albert4f378d72020-07-23 17:32:15 -07001123 platformSdkCodename: "S",
1124 activeCodenames: []string{"S"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001125 },
1126 }
1127
1128 for _, moduleType := range []string{"android_app", "android_library"} {
1129 for _, test := range testCases {
1130 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001131 platformApiProp := ""
1132 if test.platformApis {
1133 platformApiProp = "platform_apis: true,"
1134 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001135 bp := fmt.Sprintf(`%s {
1136 name: "foo",
1137 srcs: ["a.java"],
1138 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001139 %s
1140 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001141
Colin Cross98be1bb2019-12-13 20:41:13 -08001142 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001143 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1144 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
Dan Albert4f378d72020-07-23 17:32:15 -07001145 config.TestProductVariables.Platform_version_active_codenames = test.activeCodenames
Colin Crossd09b0b62018-04-18 11:06:47 -07001146 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
Jeongik Cha219141c2020-08-06 23:00:37 +09001147 checkSdkVersion(t, config, test.expectedMinSdkVersion)
Colin Crossd09b0b62018-04-18 11:06:47 -07001148
Colin Crossd09b0b62018-04-18 11:06:47 -07001149 })
1150 }
1151 }
1152}
Colin Crossa4f08812018-10-02 22:03:40 -07001153
Jeongik Cha219141c2020-08-06 23:00:37 +09001154func TestVendorAppSdkVersion(t *testing.T) {
1155 testCases := []struct {
1156 name string
1157 sdkVersion string
1158 platformSdkInt int
1159 platformSdkCodename string
1160 platformSdkFinal bool
1161 deviceCurrentApiLevelForVendorModules string
1162 expectedMinSdkVersion string
1163 }{
1164 {
1165 name: "current final SDK",
1166 sdkVersion: "current",
1167 platformSdkInt: 29,
1168 platformSdkCodename: "REL",
1169 platformSdkFinal: true,
1170 deviceCurrentApiLevelForVendorModules: "29",
1171 expectedMinSdkVersion: "29",
1172 },
1173 {
1174 name: "current final SDK",
1175 sdkVersion: "current",
1176 platformSdkInt: 29,
1177 platformSdkCodename: "REL",
1178 platformSdkFinal: true,
1179 deviceCurrentApiLevelForVendorModules: "28",
1180 expectedMinSdkVersion: "28",
1181 },
1182 {
1183 name: "current final SDK",
1184 sdkVersion: "current",
1185 platformSdkInt: 29,
1186 platformSdkCodename: "Q",
1187 platformSdkFinal: false,
Jeongik Cha219141c2020-08-06 23:00:37 +09001188 deviceCurrentApiLevelForVendorModules: "28",
1189 expectedMinSdkVersion: "28",
1190 },
1191 }
1192
1193 for _, moduleType := range []string{"android_app", "android_library"} {
1194 for _, sdkKind := range []string{"", "system_"} {
1195 for _, test := range testCases {
1196 t.Run(moduleType+" "+test.name, func(t *testing.T) {
1197 bp := fmt.Sprintf(`%s {
1198 name: "foo",
1199 srcs: ["a.java"],
1200 sdk_version: "%s%s",
1201 vendor: true,
1202 }`, moduleType, sdkKind, test.sdkVersion)
1203
1204 config := testAppConfig(nil, bp, nil)
1205 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1206 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1207 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1208 config.TestProductVariables.DeviceCurrentApiLevelForVendorModules = &test.deviceCurrentApiLevelForVendorModules
1209 config.TestProductVariables.DeviceSystemSdkVersions = []string{"28", "29"}
1210 checkSdkVersion(t, config, test.expectedMinSdkVersion)
1211 })
1212 }
1213 }
1214 }
1215}
1216
Paul Duffin50c217c2019-06-12 13:25:22 +01001217func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001218 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001219 cc_library {
1220 name: "libjni",
1221 system_shared_libs: [],
Colin Crossc511bc52020-04-07 16:50:32 +00001222 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001223 stl: "none",
1224 }
1225
1226 android_test {
1227 name: "test",
1228 sdk_version: "core_platform",
1229 jni_libs: ["libjni"],
1230 }
1231
1232 android_test {
1233 name: "test_first",
1234 sdk_version: "core_platform",
1235 compile_multilib: "first",
1236 jni_libs: ["libjni"],
1237 }
1238
1239 android_test {
1240 name: "test_both",
1241 sdk_version: "core_platform",
1242 compile_multilib: "both",
1243 jni_libs: ["libjni"],
1244 }
1245
1246 android_test {
1247 name: "test_32",
1248 sdk_version: "core_platform",
1249 compile_multilib: "32",
1250 jni_libs: ["libjni"],
1251 }
1252
1253 android_test {
1254 name: "test_64",
1255 sdk_version: "core_platform",
1256 compile_multilib: "64",
1257 jni_libs: ["libjni"],
1258 }
1259 `)
1260
1261 testCases := []struct {
1262 name string
1263 abis []string
1264 }{
1265 {"test", []string{"arm64-v8a"}},
1266 {"test_first", []string{"arm64-v8a"}},
1267 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1268 {"test_32", []string{"armeabi-v7a"}},
1269 {"test_64", []string{"arm64-v8a"}},
1270 }
1271
1272 for _, test := range testCases {
1273 t.Run(test.name, func(t *testing.T) {
1274 app := ctx.ModuleForTests(test.name, "android_common")
1275 jniLibZip := app.Output("jnilibs.zip")
1276 var abis []string
1277 args := strings.Fields(jniLibZip.Args["jarArgs"])
1278 for i := 0; i < len(args); i++ {
1279 if args[i] == "-P" {
1280 abis = append(abis, filepath.Base(args[i+1]))
1281 i++
1282 }
1283 }
1284 if !reflect.DeepEqual(abis, test.abis) {
1285 t.Errorf("want abis %v, got %v", test.abis, abis)
1286 }
1287 })
1288 }
1289}
1290
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001291func TestAppSdkVersionByPartition(t *testing.T) {
1292 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1293 android_app {
1294 name: "foo",
1295 srcs: ["a.java"],
1296 vendor: true,
1297 platform_apis: true,
1298 }
1299 `)
1300
1301 testJava(t, `
1302 android_app {
1303 name: "bar",
1304 srcs: ["b.java"],
1305 platform_apis: true,
1306 }
1307 `)
1308
1309 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001310 bp := `
1311 android_app {
1312 name: "foo",
1313 srcs: ["a.java"],
1314 product_specific: true,
1315 platform_apis: true,
1316 }
1317 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001318
1319 config := testAppConfig(nil, bp, nil)
1320 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001321 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001322 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 +09001323 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001324 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001325 }
1326 }
1327}
1328
Paul Duffin50c217c2019-06-12 13:25:22 +01001329func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001330 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001331 cc_library {
1332 name: "libjni",
1333 system_shared_libs: [],
1334 stl: "none",
Colin Cross094cde42020-02-15 10:38:00 -08001335 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001336 }
1337
1338 android_app {
1339 name: "app",
1340 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001341 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001342 }
1343
1344 android_app {
1345 name: "app_noembed",
1346 jni_libs: ["libjni"],
1347 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001348 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001349 }
1350
1351 android_app {
1352 name: "app_embed",
1353 jni_libs: ["libjni"],
1354 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001355 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001356 }
1357
1358 android_test {
1359 name: "test",
Colin Crossc511bc52020-04-07 16:50:32 +00001360 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001361 jni_libs: ["libjni"],
1362 }
1363
1364 android_test {
1365 name: "test_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001366 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001367 jni_libs: ["libjni"],
1368 use_embedded_native_libs: false,
1369 }
1370
1371 android_test_helper_app {
1372 name: "test_helper",
Colin Crossc511bc52020-04-07 16:50:32 +00001373 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001374 jni_libs: ["libjni"],
1375 }
1376
1377 android_test_helper_app {
1378 name: "test_helper_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001379 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001380 jni_libs: ["libjni"],
1381 use_embedded_native_libs: false,
1382 }
1383 `)
1384
1385 testCases := []struct {
1386 name string
1387 packaged bool
1388 compressed bool
1389 }{
1390 {"app", false, false},
1391 {"app_noembed", false, false},
1392 {"app_embed", true, false},
1393 {"test", true, false},
1394 {"test_noembed", true, true},
1395 {"test_helper", true, false},
1396 {"test_helper_noembed", true, true},
1397 }
1398
1399 for _, test := range testCases {
1400 t.Run(test.name, func(t *testing.T) {
1401 app := ctx.ModuleForTests(test.name, "android_common")
1402 jniLibZip := app.MaybeOutput("jnilibs.zip")
1403 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1404 t.Errorf("expected jni packaged %v, got %v", w, g)
1405 }
1406
1407 if jniLibZip.Rule != nil {
1408 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1409 t.Errorf("expected jni compressed %v, got %v", w, g)
1410 }
Colin Crossc511bc52020-04-07 16:50:32 +00001411
1412 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1413 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1414 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001415 }
1416 })
1417 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001418}
1419
Colin Cross3c007702020-05-08 11:20:24 -07001420func TestJNISDK(t *testing.T) {
1421 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1422 cc_library {
1423 name: "libjni",
1424 system_shared_libs: [],
1425 stl: "none",
1426 sdk_version: "current",
1427 }
1428
1429 android_test {
1430 name: "app_platform",
1431 jni_libs: ["libjni"],
1432 platform_apis: true,
1433 }
1434
1435 android_test {
1436 name: "app_sdk",
1437 jni_libs: ["libjni"],
1438 sdk_version: "current",
1439 }
1440
1441 android_test {
1442 name: "app_force_platform",
1443 jni_libs: ["libjni"],
1444 sdk_version: "current",
1445 jni_uses_platform_apis: true,
1446 }
1447
1448 android_test {
1449 name: "app_force_sdk",
1450 jni_libs: ["libjni"],
1451 platform_apis: true,
1452 jni_uses_sdk_apis: true,
1453 }
Colin Crossc2d24052020-05-13 11:05:02 -07001454
1455 cc_library {
1456 name: "libvendorjni",
1457 system_shared_libs: [],
1458 stl: "none",
1459 vendor: true,
1460 }
1461
1462 android_test {
1463 name: "app_vendor",
1464 jni_libs: ["libvendorjni"],
1465 sdk_version: "current",
1466 vendor: true,
1467 }
Colin Cross3c007702020-05-08 11:20:24 -07001468 `)
1469
1470 testCases := []struct {
Colin Crossc2d24052020-05-13 11:05:02 -07001471 name string
1472 sdkJNI bool
1473 vendorJNI bool
Colin Cross3c007702020-05-08 11:20:24 -07001474 }{
Colin Crossc2d24052020-05-13 11:05:02 -07001475 {name: "app_platform"},
1476 {name: "app_sdk", sdkJNI: true},
1477 {name: "app_force_platform"},
1478 {name: "app_force_sdk", sdkJNI: true},
1479 {name: "app_vendor", vendorJNI: true},
Colin Cross3c007702020-05-08 11:20:24 -07001480 }
1481
Colin Crossc2d24052020-05-13 11:05:02 -07001482 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1483 Output("libjni.so").Output.String()
1484 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1485 Output("libjni.so").Output.String()
1486 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1487 Output("libvendorjni.so").Output.String()
1488
Colin Cross3c007702020-05-08 11:20:24 -07001489 for _, test := range testCases {
1490 t.Run(test.name, func(t *testing.T) {
1491 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross3c007702020-05-08 11:20:24 -07001492
1493 jniLibZip := app.MaybeOutput("jnilibs.zip")
1494 if len(jniLibZip.Implicits) != 1 {
1495 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1496 }
1497 gotJNI := jniLibZip.Implicits[0].String()
1498
1499 if test.sdkJNI {
1500 if gotJNI != sdkJNI {
1501 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1502 }
Colin Crossc2d24052020-05-13 11:05:02 -07001503 } else if test.vendorJNI {
1504 if gotJNI != vendorJNI {
1505 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1506 }
Colin Cross3c007702020-05-08 11:20:24 -07001507 } else {
1508 if gotJNI != platformJNI {
1509 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1510 }
1511 }
1512 })
1513 }
1514
1515 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1516 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1517 android_test {
1518 name: "app_platform",
1519 platform_apis: true,
1520 jni_uses_platform_apis: true,
1521 }
1522 `)
1523 })
1524
1525 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1526 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1527 android_test {
1528 name: "app_sdk",
1529 sdk_version: "current",
1530 jni_uses_sdk_apis: true,
1531 }
1532 `)
1533 })
1534
1535}
1536
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001537func TestCertificates(t *testing.T) {
1538 testCases := []struct {
1539 name string
1540 bp string
1541 certificateOverride string
Liz Kammere2b27f42020-05-07 13:24:05 -07001542 expectedLineage string
1543 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001544 }{
1545 {
1546 name: "default",
1547 bp: `
1548 android_app {
1549 name: "foo",
1550 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001551 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001552 }
1553 `,
1554 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001555 expectedLineage: "",
1556 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001557 },
1558 {
1559 name: "module certificate property",
1560 bp: `
1561 android_app {
1562 name: "foo",
1563 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001564 certificate: ":new_certificate",
1565 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001566 }
1567
1568 android_app_certificate {
1569 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001570 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001571 }
1572 `,
1573 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001574 expectedLineage: "",
1575 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001576 },
1577 {
1578 name: "path certificate property",
1579 bp: `
1580 android_app {
1581 name: "foo",
1582 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001583 certificate: "expiredkey",
1584 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001585 }
1586 `,
1587 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001588 expectedLineage: "",
1589 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001590 },
1591 {
1592 name: "certificate overrides",
1593 bp: `
1594 android_app {
1595 name: "foo",
1596 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001597 certificate: "expiredkey",
1598 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001599 }
1600
1601 android_app_certificate {
1602 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001603 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001604 }
1605 `,
1606 certificateOverride: "foo:new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001607 expectedLineage: "",
1608 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1609 },
1610 {
1611 name: "certificate lineage",
1612 bp: `
1613 android_app {
1614 name: "foo",
1615 srcs: ["a.java"],
1616 certificate: ":new_certificate",
1617 lineage: "lineage.bin",
1618 sdk_version: "current",
1619 }
1620
1621 android_app_certificate {
1622 name: "new_certificate",
1623 certificate: "cert/new_cert",
1624 }
1625 `,
1626 certificateOverride: "",
1627 expectedLineage: "--lineage lineage.bin",
1628 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001629 },
1630 }
1631
1632 for _, test := range testCases {
1633 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001634 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001635 if test.certificateOverride != "" {
1636 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1637 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001638 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001639
1640 run(t, ctx, config)
1641 foo := ctx.ModuleForTests("foo", "android_common")
1642
1643 signapk := foo.Output("foo.apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001644 signCertificateFlags := signapk.Args["certificates"]
1645 if test.expectedCertificate != signCertificateFlags {
1646 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1647 }
1648
1649 signFlags := signapk.Args["flags"]
1650 if test.expectedLineage != signFlags {
1651 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001652 }
1653 })
1654 }
1655}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001656
Songchun Fan688de9a2020-03-24 20:32:24 -07001657func TestRequestV4SigningFlag(t *testing.T) {
1658 testCases := []struct {
1659 name string
1660 bp string
1661 expected string
1662 }{
1663 {
1664 name: "default",
1665 bp: `
1666 android_app {
1667 name: "foo",
1668 srcs: ["a.java"],
1669 sdk_version: "current",
1670 }
1671 `,
1672 expected: "",
1673 },
1674 {
1675 name: "default",
1676 bp: `
1677 android_app {
1678 name: "foo",
1679 srcs: ["a.java"],
1680 sdk_version: "current",
1681 v4_signature: false,
1682 }
1683 `,
1684 expected: "",
1685 },
1686 {
1687 name: "module certificate property",
1688 bp: `
1689 android_app {
1690 name: "foo",
1691 srcs: ["a.java"],
1692 sdk_version: "current",
1693 v4_signature: true,
1694 }
1695 `,
1696 expected: "--enable-v4",
1697 },
1698 }
1699
1700 for _, test := range testCases {
1701 t.Run(test.name, func(t *testing.T) {
1702 config := testAppConfig(nil, test.bp, nil)
1703 ctx := testContext()
1704
1705 run(t, ctx, config)
1706 foo := ctx.ModuleForTests("foo", "android_common")
1707
1708 signapk := foo.Output("foo.apk")
1709 signFlags := signapk.Args["flags"]
1710 if test.expected != signFlags {
1711 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1712 }
1713 })
1714 }
1715}
1716
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001717func TestPackageNameOverride(t *testing.T) {
1718 testCases := []struct {
1719 name string
1720 bp string
1721 packageNameOverride string
1722 expected []string
1723 }{
1724 {
1725 name: "default",
1726 bp: `
1727 android_app {
1728 name: "foo",
1729 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001730 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001731 }
1732 `,
1733 packageNameOverride: "",
1734 expected: []string{
1735 buildDir + "/.intermediates/foo/android_common/foo.apk",
1736 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1737 },
1738 },
1739 {
1740 name: "overridden",
1741 bp: `
1742 android_app {
1743 name: "foo",
1744 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001745 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001746 }
1747 `,
1748 packageNameOverride: "foo:bar",
1749 expected: []string{
1750 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001751 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001752 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1753 },
1754 },
1755 }
1756
1757 for _, test := range testCases {
1758 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001759 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001760 if test.packageNameOverride != "" {
1761 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1762 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001763 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001764
1765 run(t, ctx, config)
1766 foo := ctx.ModuleForTests("foo", "android_common")
1767
1768 outputs := foo.AllOutputs()
1769 outputMap := make(map[string]bool)
1770 for _, o := range outputs {
1771 outputMap[o] = true
1772 }
1773 for _, e := range test.expected {
1774 if _, exist := outputMap[e]; !exist {
1775 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1776 }
1777 }
1778 })
1779 }
1780}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001781
1782func TestInstrumentationTargetOverridden(t *testing.T) {
1783 bp := `
1784 android_app {
1785 name: "foo",
1786 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001787 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001788 }
1789
1790 android_test {
1791 name: "bar",
1792 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001793 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001794 }
1795 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001796 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001797 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001798 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001799
1800 run(t, ctx, config)
1801
1802 bar := ctx.ModuleForTests("bar", "android_common")
1803 res := bar.Output("package-res.apk")
1804 aapt2Flags := res.Args["flags"]
1805 e := "--rename-instrumentation-target-package org.dandroid.bp"
1806 if !strings.Contains(aapt2Flags, e) {
1807 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1808 }
1809}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001810
1811func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001812 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001813 android_app {
1814 name: "foo",
1815 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001816 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001817 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001818 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001819 }
1820
1821 override_android_app {
1822 name: "bar",
1823 base: "foo",
1824 certificate: ":new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001825 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001826 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001827 }
1828
1829 android_app_certificate {
1830 name: "new_certificate",
1831 certificate: "cert/new_cert",
1832 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001833
1834 override_android_app {
1835 name: "baz",
1836 base: "foo",
1837 package_name: "org.dandroid.bp",
1838 }
Liz Kammer9f9fd022020-06-18 19:44:06 +00001839
1840 override_android_app {
1841 name: "baz_no_rename_resources",
1842 base: "foo",
1843 package_name: "org.dandroid.bp",
1844 rename_resources_package: false,
1845 }
1846
1847 android_app {
1848 name: "foo_no_rename_resources",
1849 srcs: ["a.java"],
1850 certificate: "expiredkey",
1851 overrides: ["qux"],
1852 rename_resources_package: false,
1853 sdk_version: "current",
1854 }
1855
1856 override_android_app {
1857 name: "baz_base_no_rename_resources",
1858 base: "foo_no_rename_resources",
1859 package_name: "org.dandroid.bp",
1860 }
1861
1862 override_android_app {
1863 name: "baz_override_base_rename_resources",
1864 base: "foo_no_rename_resources",
1865 package_name: "org.dandroid.bp",
1866 rename_resources_package: true,
1867 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001868 `)
1869
1870 expectedVariants := []struct {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001871 name string
1872 moduleName string
1873 variantName string
1874 apkName string
1875 apkPath string
1876 certFlag string
1877 lineageFlag string
1878 overrides []string
1879 packageFlag string
1880 renameResources bool
1881 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001882 }{
1883 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001884 name: "foo",
1885 moduleName: "foo",
1886 variantName: "android_common",
1887 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
1888 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1889 lineageFlag: "",
1890 overrides: []string{"qux"},
1891 packageFlag: "",
1892 renameResources: false,
1893 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001894 },
1895 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001896 name: "foo",
1897 moduleName: "bar",
1898 variantName: "android_common_bar",
1899 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
1900 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1901 lineageFlag: "--lineage lineage.bin",
1902 overrides: []string{"qux", "foo"},
1903 packageFlag: "",
1904 renameResources: false,
1905 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001906 },
1907 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001908 name: "foo",
1909 moduleName: "baz",
1910 variantName: "android_common_baz",
1911 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
1912 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1913 lineageFlag: "",
1914 overrides: []string{"qux", "foo"},
1915 packageFlag: "org.dandroid.bp",
1916 renameResources: true,
1917 logging_parent: "",
1918 },
1919 {
1920 name: "foo",
1921 moduleName: "baz_no_rename_resources",
1922 variantName: "android_common_baz_no_rename_resources",
1923 apkPath: "/target/product/test_device/system/app/baz_no_rename_resources/baz_no_rename_resources.apk",
1924 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1925 lineageFlag: "",
1926 overrides: []string{"qux", "foo"},
1927 packageFlag: "org.dandroid.bp",
1928 renameResources: false,
1929 logging_parent: "",
1930 },
1931 {
1932 name: "foo_no_rename_resources",
1933 moduleName: "baz_base_no_rename_resources",
1934 variantName: "android_common_baz_base_no_rename_resources",
1935 apkPath: "/target/product/test_device/system/app/baz_base_no_rename_resources/baz_base_no_rename_resources.apk",
1936 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1937 lineageFlag: "",
1938 overrides: []string{"qux", "foo_no_rename_resources"},
1939 packageFlag: "org.dandroid.bp",
1940 renameResources: false,
1941 logging_parent: "",
1942 },
1943 {
1944 name: "foo_no_rename_resources",
1945 moduleName: "baz_override_base_rename_resources",
1946 variantName: "android_common_baz_override_base_rename_resources",
1947 apkPath: "/target/product/test_device/system/app/baz_override_base_rename_resources/baz_override_base_rename_resources.apk",
1948 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1949 lineageFlag: "",
1950 overrides: []string{"qux", "foo_no_rename_resources"},
1951 packageFlag: "org.dandroid.bp",
1952 renameResources: true,
1953 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001954 },
1955 }
1956 for _, expected := range expectedVariants {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001957 variant := ctx.ModuleForTests(expected.name, expected.variantName)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001958
1959 // Check the final apk name
1960 outputs := variant.AllOutputs()
1961 expectedApkPath := buildDir + expected.apkPath
1962 found := false
1963 for _, o := range outputs {
1964 if o == expectedApkPath {
1965 found = true
1966 break
1967 }
1968 }
1969 if !found {
1970 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1971 }
1972
1973 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001974 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001975 certFlag := signapk.Args["certificates"]
1976 if expected.certFlag != certFlag {
1977 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1978 }
1979
1980 // Check the lineage flags
1981 lineageFlag := signapk.Args["flags"]
1982 if expected.lineageFlag != lineageFlag {
1983 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001984 }
1985
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001986 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001987 mod := variant.Module().(*AndroidApp)
1988 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1989 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1990 expected.overrides, mod.appProperties.Overrides)
1991 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001992
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001993 // Test Overridable property: Logging_parent
1994 logging_parent := mod.aapt.LoggingParent
1995 if expected.logging_parent != logging_parent {
1996 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1997 expected.logging_parent, logging_parent)
1998 }
1999
Liz Kammer1d5983b2020-05-19 19:15:37 +00002000 // Check the package renaming flag, if exists.
Jaewoong Jung6f373f62019-03-13 10:13:24 -07002001 res := variant.Output("package-res.apk")
2002 aapt2Flags := res.Args["flags"]
Liz Kammer9f9fd022020-06-18 19:44:06 +00002003 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
2004 expectedPackage := expected.packageFlag
2005 if !expected.renameResources {
2006 expectedPackage = ""
Liz Kammer1d5983b2020-05-19 19:15:37 +00002007 }
Liz Kammer9f9fd022020-06-18 19:44:06 +00002008 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expectedPackage)
Jaewoong Jung525443a2019-02-28 15:35:54 -08002009 }
2010}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002011
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07002012func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002013 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07002014 android_app {
2015 name: "foo",
2016 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002017 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07002018 }
2019
2020 override_android_app {
2021 name: "bar",
2022 base: "foo",
2023 package_name: "org.dandroid.bp",
2024 }
2025
2026 android_test {
2027 name: "baz",
2028 srcs: ["b.java"],
2029 instrumentation_for: "foo",
2030 }
2031
2032 android_test {
2033 name: "qux",
2034 srcs: ["b.java"],
2035 instrumentation_for: "bar",
2036 }
2037 `)
2038
2039 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
2040 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
2041 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
2042 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
2043 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
2044 }
2045
2046 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
2047 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
2048 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
2049 if !strings.Contains(javac.Args["classpath"], barTurbine) {
2050 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
2051 }
2052}
2053
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002054func TestOverrideAndroidTest(t *testing.T) {
2055 ctx, _ := testJava(t, `
2056 android_app {
2057 name: "foo",
2058 srcs: ["a.java"],
2059 package_name: "com.android.foo",
2060 sdk_version: "current",
2061 }
2062
2063 override_android_app {
2064 name: "bar",
2065 base: "foo",
2066 package_name: "com.android.bar",
2067 }
2068
2069 android_test {
2070 name: "foo_test",
2071 srcs: ["b.java"],
2072 instrumentation_for: "foo",
2073 }
2074
2075 override_android_test {
2076 name: "bar_test",
2077 base: "foo_test",
2078 package_name: "com.android.bar.test",
2079 instrumentation_for: "bar",
2080 instrumentation_target_package: "com.android.bar",
2081 }
2082 `)
2083
2084 expectedVariants := []struct {
2085 moduleName string
2086 variantName string
2087 apkPath string
2088 overrides []string
2089 targetVariant string
2090 packageFlag string
2091 targetPackageFlag string
2092 }{
2093 {
2094 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002095 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002096 overrides: nil,
2097 targetVariant: "android_common",
2098 packageFlag: "",
2099 targetPackageFlag: "",
2100 },
2101 {
2102 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002103 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002104 overrides: []string{"foo_test"},
2105 targetVariant: "android_common_bar",
2106 packageFlag: "com.android.bar.test",
2107 targetPackageFlag: "com.android.bar",
2108 },
2109 }
2110 for _, expected := range expectedVariants {
2111 variant := ctx.ModuleForTests("foo_test", expected.variantName)
2112
2113 // Check the final apk name
2114 outputs := variant.AllOutputs()
2115 expectedApkPath := buildDir + expected.apkPath
2116 found := false
2117 for _, o := range outputs {
2118 if o == expectedApkPath {
2119 found = true
2120 break
2121 }
2122 }
2123 if !found {
2124 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
2125 }
2126
2127 // Check if the overrides field values are correctly aggregated.
2128 mod := variant.Module().(*AndroidTest)
2129 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
2130 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
2131 expected.overrides, mod.appProperties.Overrides)
2132 }
2133
2134 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
2135 javac := variant.Rule("javac")
2136 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
2137 if !strings.Contains(javac.Args["classpath"], turbine) {
2138 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
2139 }
2140
2141 // Check aapt2 flags.
2142 res := variant.Output("package-res.apk")
2143 aapt2Flags := res.Args["flags"]
2144 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammer9f9fd022020-06-18 19:44:06 +00002145 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expected.packageFlag)
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002146 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
2147 }
2148}
2149
Jaewoong Jung39982342020-01-14 10:27:18 -08002150func TestAndroidTest_FixTestConfig(t *testing.T) {
2151 ctx, _ := testJava(t, `
2152 android_app {
2153 name: "foo",
2154 srcs: ["a.java"],
2155 package_name: "com.android.foo",
2156 sdk_version: "current",
2157 }
2158
2159 android_test {
2160 name: "foo_test",
2161 srcs: ["b.java"],
2162 instrumentation_for: "foo",
2163 }
2164
2165 android_test {
2166 name: "bar_test",
2167 srcs: ["b.java"],
2168 package_name: "com.android.bar.test",
2169 instrumentation_for: "foo",
2170 }
2171
2172 override_android_test {
2173 name: "baz_test",
2174 base: "foo_test",
2175 package_name: "com.android.baz.test",
2176 }
2177 `)
2178
2179 testCases := []struct {
2180 moduleName string
2181 variantName string
2182 expectedFlags []string
2183 }{
2184 {
2185 moduleName: "foo_test",
2186 variantName: "android_common",
2187 },
2188 {
2189 moduleName: "bar_test",
2190 variantName: "android_common",
2191 expectedFlags: []string{
2192 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
2193 "--package-name com.android.bar.test",
2194 },
2195 },
2196 {
2197 moduleName: "foo_test",
2198 variantName: "android_common_baz_test",
2199 expectedFlags: []string{
2200 "--manifest " + buildDir +
2201 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
2202 "--package-name com.android.baz.test",
2203 "--test-file-name baz_test.apk",
2204 },
2205 },
2206 }
2207
2208 for _, test := range testCases {
2209 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
2210 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
2211
2212 if len(test.expectedFlags) > 0 {
2213 if params.Rule == nil {
2214 t.Errorf("test_config_fixer was expected to run, but didn't")
2215 } else {
2216 for _, flag := range test.expectedFlags {
2217 if !strings.Contains(params.RuleParams.Command, flag) {
2218 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2219 }
2220 }
2221 }
2222 } else {
2223 if params.Rule != nil {
2224 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2225 }
2226 }
2227
2228 }
2229}
2230
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002231func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002232 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002233 android_app_import {
2234 name: "foo",
2235 apk: "prebuilts/apk/app.apk",
2236 certificate: "platform",
2237 dex_preopt: {
2238 enabled: true,
2239 },
2240 }
2241 `)
2242
2243 variant := ctx.ModuleForTests("foo", "android_common")
2244
2245 // Check dexpreopt outputs.
2246 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2247 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2248 t.Errorf("can't find dexpreopt outputs")
2249 }
2250
2251 // Check cert signing flag.
2252 signedApk := variant.Output("signed/foo.apk")
2253 signingFlag := signedApk.Args["certificates"]
2254 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2255 if expected != signingFlag {
2256 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2257 }
2258}
2259
2260func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002261 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002262 android_app_import {
2263 name: "foo",
2264 apk: "prebuilts/apk/app.apk",
2265 certificate: "platform",
2266 dex_preopt: {
2267 enabled: false,
2268 },
2269 }
2270 `)
2271
2272 variant := ctx.ModuleForTests("foo", "android_common")
2273
2274 // Check dexpreopt outputs. They shouldn't exist.
2275 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2276 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2277 t.Errorf("dexpreopt shouldn't have run.")
2278 }
2279}
2280
2281func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002282 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002283 android_app_import {
2284 name: "foo",
2285 apk: "prebuilts/apk/app.apk",
2286 presigned: true,
2287 dex_preopt: {
2288 enabled: true,
2289 },
2290 }
2291 `)
2292
2293 variant := ctx.ModuleForTests("foo", "android_common")
2294
2295 // Check dexpreopt outputs.
2296 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2297 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2298 t.Errorf("can't find dexpreopt outputs")
2299 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002300 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002301 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2302 t.Errorf("signing rule shouldn't be included.")
2303 }
2304 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2305 t.Errorf("can't find aligning rule")
2306 }
2307}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002308
Liz Kammer24978992020-05-13 15:49:21 -07002309func TestAndroidAppImport_SigningLineage(t *testing.T) {
2310 ctx, _ := testJava(t, `
2311 android_app_import {
2312 name: "foo",
2313 apk: "prebuilts/apk/app.apk",
2314 certificate: "platform",
2315 lineage: "lineage.bin",
2316 }
2317 `)
2318
2319 variant := ctx.ModuleForTests("foo", "android_common")
2320
2321 // Check cert signing lineage flag.
2322 signedApk := variant.Output("signed/foo.apk")
2323 signingFlag := signedApk.Args["flags"]
2324 expected := "--lineage lineage.bin"
2325 if expected != signingFlag {
2326 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2327 }
2328}
2329
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002330func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2331 ctx, _ := testJava(t, `
2332 android_app_import {
2333 name: "foo",
2334 apk: "prebuilts/apk/app.apk",
2335 default_dev_cert: true,
2336 dex_preopt: {
2337 enabled: true,
2338 },
2339 }
2340 `)
2341
2342 variant := ctx.ModuleForTests("foo", "android_common")
2343
2344 // Check dexpreopt outputs.
2345 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2346 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2347 t.Errorf("can't find dexpreopt outputs")
2348 }
2349
2350 // Check cert signing flag.
2351 signedApk := variant.Output("signed/foo.apk")
2352 signingFlag := signedApk.Args["certificates"]
2353 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2354 if expected != signingFlag {
2355 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2356 }
2357}
2358
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002359func TestAndroidAppImport_DpiVariants(t *testing.T) {
2360 bp := `
2361 android_app_import {
2362 name: "foo",
2363 apk: "prebuilts/apk/app.apk",
2364 dpi_variants: {
2365 xhdpi: {
2366 apk: "prebuilts/apk/app_xhdpi.apk",
2367 },
2368 xxhdpi: {
2369 apk: "prebuilts/apk/app_xxhdpi.apk",
2370 },
2371 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002372 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002373 dex_preopt: {
2374 enabled: true,
2375 },
2376 }
2377 `
2378 testCases := []struct {
2379 name string
2380 aaptPreferredConfig *string
2381 aaptPrebuiltDPI []string
2382 expected string
2383 }{
2384 {
2385 name: "no preferred",
2386 aaptPreferredConfig: nil,
2387 aaptPrebuiltDPI: []string{},
2388 expected: "prebuilts/apk/app.apk",
2389 },
2390 {
2391 name: "AAPTPreferredConfig matches",
2392 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002393 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002394 expected: "prebuilts/apk/app_xhdpi.apk",
2395 },
2396 {
2397 name: "AAPTPrebuiltDPI matches",
2398 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2399 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2400 expected: "prebuilts/apk/app_xxhdpi.apk",
2401 },
2402 {
2403 name: "non-first AAPTPrebuiltDPI matches",
2404 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2405 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2406 expected: "prebuilts/apk/app_xhdpi.apk",
2407 },
2408 {
2409 name: "no matches",
2410 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2411 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2412 expected: "prebuilts/apk/app.apk",
2413 },
2414 }
2415
2416 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2417 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002418 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002419 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2420 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002421 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002422
2423 run(t, ctx, config)
2424
2425 variant := ctx.ModuleForTests("foo", "android_common")
2426 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2427 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2428 if len(matches) != 2 {
2429 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2430 }
2431 if test.expected != matches[1] {
2432 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2433 }
2434 }
2435}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002436
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002437func TestAndroidAppImport_Filename(t *testing.T) {
2438 ctx, config := testJava(t, `
2439 android_app_import {
2440 name: "foo",
2441 apk: "prebuilts/apk/app.apk",
2442 presigned: true,
2443 }
2444
2445 android_app_import {
2446 name: "bar",
2447 apk: "prebuilts/apk/app.apk",
2448 presigned: true,
2449 filename: "bar_sample.apk"
2450 }
2451 `)
2452
2453 testCases := []struct {
2454 name string
2455 expected string
2456 }{
2457 {
2458 name: "foo",
2459 expected: "foo.apk",
2460 },
2461 {
2462 name: "bar",
2463 expected: "bar_sample.apk",
2464 },
2465 }
2466
2467 for _, test := range testCases {
2468 variant := ctx.ModuleForTests(test.name, "android_common")
2469 if variant.MaybeOutput(test.expected).Rule == nil {
2470 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2471 }
2472
2473 a := variant.Module().(*AndroidAppImport)
2474 expectedValues := []string{test.expected}
2475 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002476 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002477 if !reflect.DeepEqual(actualValues, expectedValues) {
2478 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2479 actualValues, expectedValues)
2480 }
2481 }
2482}
2483
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002484func TestAndroidAppImport_ArchVariants(t *testing.T) {
2485 // The test config's target arch is ARM64.
2486 testCases := []struct {
2487 name string
2488 bp string
2489 expected string
2490 }{
2491 {
2492 name: "matching arch",
2493 bp: `
2494 android_app_import {
2495 name: "foo",
2496 apk: "prebuilts/apk/app.apk",
2497 arch: {
2498 arm64: {
2499 apk: "prebuilts/apk/app_arm64.apk",
2500 },
2501 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002502 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002503 dex_preopt: {
2504 enabled: true,
2505 },
2506 }
2507 `,
2508 expected: "prebuilts/apk/app_arm64.apk",
2509 },
2510 {
2511 name: "no matching arch",
2512 bp: `
2513 android_app_import {
2514 name: "foo",
2515 apk: "prebuilts/apk/app.apk",
2516 arch: {
2517 arm: {
2518 apk: "prebuilts/apk/app_arm.apk",
2519 },
2520 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002521 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002522 dex_preopt: {
2523 enabled: true,
2524 },
2525 }
2526 `,
2527 expected: "prebuilts/apk/app.apk",
2528 },
2529 }
2530
2531 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2532 for _, test := range testCases {
2533 ctx, _ := testJava(t, test.bp)
2534
2535 variant := ctx.ModuleForTests("foo", "android_common")
2536 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2537 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2538 if len(matches) != 2 {
2539 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2540 }
2541 if test.expected != matches[1] {
2542 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2543 }
2544 }
2545}
2546
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002547func TestAndroidTestImport(t *testing.T) {
2548 ctx, config := testJava(t, `
2549 android_test_import {
2550 name: "foo",
2551 apk: "prebuilts/apk/app.apk",
2552 presigned: true,
2553 data: [
2554 "testdata/data",
2555 ],
2556 }
2557 `)
2558
2559 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2560
2561 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002562 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002563 expected := []string{"tests"}
2564 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2565 if !reflect.DeepEqual(expected, actual) {
2566 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2567 }
2568 expected = []string{"testdata/data:testdata/data"}
2569 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2570 if !reflect.DeepEqual(expected, actual) {
2571 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2572 }
2573}
2574
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002575func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2576 ctx, _ := testJava(t, `
2577 android_test_import {
2578 name: "foo",
2579 apk: "prebuilts/apk/app.apk",
2580 certificate: "cert/new_cert",
2581 data: [
2582 "testdata/data",
2583 ],
2584 }
2585
2586 android_test_import {
2587 name: "foo_presigned",
2588 apk: "prebuilts/apk/app.apk",
2589 presigned: true,
2590 data: [
2591 "testdata/data",
2592 ],
2593 }
2594 `)
2595
2596 variant := ctx.ModuleForTests("foo", "android_common")
2597 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2598 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2599 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2600 }
2601
2602 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2603 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2604 if jniRule != android.Cp.String() {
2605 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2606 }
Liz Kammer3b70b3f2020-05-20 14:36:30 -07002607 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2608 t.Errorf("Presigned test apk should be aligned")
2609 }
2610}
2611
2612func TestAndroidTestImport_Preprocessed(t *testing.T) {
2613 ctx, _ := testJava(t, `
2614 android_test_import {
2615 name: "foo",
2616 apk: "prebuilts/apk/app.apk",
2617 presigned: true,
2618 preprocessed: true,
2619 }
2620
2621 android_test_import {
2622 name: "foo_cert",
2623 apk: "prebuilts/apk/app.apk",
2624 certificate: "cert/new_cert",
2625 preprocessed: true,
2626 }
2627 `)
2628
2629 testModules := []string{"foo", "foo_cert"}
2630 for _, m := range testModules {
2631 apkName := m + ".apk"
2632 variant := ctx.ModuleForTests(m, "android_common")
2633 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2634 if jniRule != android.Cp.String() {
2635 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2636 }
2637
2638 // Make sure signing and aligning were skipped.
2639 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2640 t.Errorf("signing rule shouldn't be included for preprocessed.")
2641 }
2642 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2643 t.Errorf("aligning rule shouldn't be for preprocessed")
2644 }
2645 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002646}
2647
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002648func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002649 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002650 cc_library {
2651 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002652 sdk_version: "current",
2653 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002654 }
2655
2656 android_test {
2657 name: "stl",
2658 jni_libs: ["libjni"],
2659 compile_multilib: "both",
2660 sdk_version: "current",
2661 stl: "c++_shared",
2662 }
2663
2664 android_test {
2665 name: "system",
2666 jni_libs: ["libjni"],
2667 compile_multilib: "both",
2668 sdk_version: "current",
2669 }
2670 `)
2671
2672 testCases := []struct {
2673 name string
2674 jnis []string
2675 }{
2676 {"stl",
2677 []string{
2678 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002679 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002680 },
2681 },
2682 {"system",
2683 []string{
2684 "libjni.so",
2685 },
2686 },
2687 }
2688
2689 for _, test := range testCases {
2690 t.Run(test.name, func(t *testing.T) {
2691 app := ctx.ModuleForTests(test.name, "android_common")
2692 jniLibZip := app.Output("jnilibs.zip")
2693 var jnis []string
2694 args := strings.Fields(jniLibZip.Args["jarArgs"])
2695 for i := 0; i < len(args); i++ {
2696 if args[i] == "-f" {
2697 jnis = append(jnis, args[i+1])
2698 i += 1
2699 }
2700 }
2701 jnisJoined := strings.Join(jnis, " ")
2702 for _, jni := range test.jnis {
2703 if !strings.Contains(jnisJoined, jni) {
2704 t.Errorf("missing jni %q in %q", jni, jnis)
2705 }
2706 }
2707 })
2708 }
2709}
Colin Cross50ddcc42019-05-16 12:28:22 -07002710
2711func TestUsesLibraries(t *testing.T) {
2712 bp := `
2713 java_sdk_library {
2714 name: "foo",
2715 srcs: ["a.java"],
2716 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002717 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002718 }
2719
2720 java_sdk_library {
Paul Duffin859fe962020-05-15 10:20:31 +01002721 name: "qux",
2722 srcs: ["a.java"],
2723 api_packages: ["qux"],
2724 sdk_version: "current",
2725 }
2726
2727 java_sdk_library {
2728 name: "quuz",
2729 srcs: ["a.java"],
2730 api_packages: ["quuz"],
2731 sdk_version: "current",
2732 }
2733
2734 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002735 name: "bar",
2736 srcs: ["a.java"],
2737 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002738 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002739 }
2740
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002741 java_sdk_library {
2742 name: "runtime-library",
2743 srcs: ["a.java"],
2744 sdk_version: "current",
2745 }
2746
2747 java_library {
2748 name: "static-runtime-helper",
2749 srcs: ["a.java"],
2750 libs: ["runtime-library"],
2751 sdk_version: "current",
2752 }
2753
Colin Cross50ddcc42019-05-16 12:28:22 -07002754 android_app {
2755 name: "app",
2756 srcs: ["a.java"],
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002757 libs: ["qux", "quuz"],
2758 static_libs: ["static-runtime-helper"],
2759 uses_libs: ["foo"],
2760 sdk_version: "current",
2761 optional_uses_libs: [
2762 "bar",
2763 "baz",
2764 ],
2765 }
2766
2767 android_app {
2768 name: "app_with_stub_deps",
2769 srcs: ["a.java"],
Paul Duffin859fe962020-05-15 10:20:31 +01002770 libs: ["qux", "quuz.stubs"],
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002771 static_libs: ["static-runtime-helper"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002772 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002773 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002774 optional_uses_libs: [
2775 "bar",
2776 "baz",
2777 ],
2778 }
2779
2780 android_app_import {
2781 name: "prebuilt",
2782 apk: "prebuilts/apk/app.apk",
2783 certificate: "platform",
2784 uses_libs: ["foo"],
2785 optional_uses_libs: [
2786 "bar",
2787 "baz",
2788 ],
2789 }
2790 `
2791
Colin Cross98be1bb2019-12-13 20:41:13 -08002792 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002793 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2794
Colin Cross98be1bb2019-12-13 20:41:13 -08002795 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002796
2797 run(t, ctx, config)
2798
2799 app := ctx.ModuleForTests("app", "android_common")
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002800 appWithStubDeps := ctx.ModuleForTests("app_with_stub_deps", "android_common")
Colin Cross50ddcc42019-05-16 12:28:22 -07002801 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2802
Paul Duffin859fe962020-05-15 10:20:31 +01002803 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2804 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002805 for _, w := range []string{"qux", "quuz", "runtime-library"} {
2806 if !strings.Contains(manifestFixerArgs, "--uses-library "+w) {
2807 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2808 }
Paul Duffin859fe962020-05-15 10:20:31 +01002809 }
2810
Colin Cross50ddcc42019-05-16 12:28:22 -07002811 // Test that all libraries are verified
2812 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2813 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2814 t.Errorf("wanted %q in %q", w, cmd)
2815 }
2816
2817 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2818 t.Errorf("wanted %q in %q", w, cmd)
2819 }
2820
2821 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2822
2823 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2824 t.Errorf("wanted %q in %q", w, cmd)
2825 }
2826
2827 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2828 t.Errorf("wanted %q in %q", w, cmd)
2829 }
2830
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002831 // Test that all present libraries are preopted, including implicit SDK dependencies
Colin Cross50ddcc42019-05-16 12:28:22 -07002832 cmd = app.Rule("dexpreopt").RuleParams.Command
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002833 w := `--target-classpath-for-sdk any` +
2834 ` /system/framework/foo.jar` +
2835 `:/system/framework/quuz.jar` +
2836 `:/system/framework/qux.jar` +
2837 `:/system/framework/runtime-library.jar` +
2838 `:/system/framework/bar.jar`
2839 if !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002840 t.Errorf("wanted %q in %q", w, cmd)
2841 }
2842
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002843 // TODO(skvadrik) fix dexpreopt for stub libraries for which the implementation is present
2844 if appWithStubDeps.MaybeRule("dexpreopt").RuleParams.Command != "" {
2845 t.Errorf("dexpreopt should be disabled for apps with dependencies on stub libraries")
2846 }
Colin Cross50ddcc42019-05-16 12:28:22 -07002847
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002848 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
Ulya Trafimovich5f364b62020-06-30 12:39:01 +01002849 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 -07002850 t.Errorf("wanted %q in %q", w, cmd)
2851 }
2852}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002853
2854func TestCodelessApp(t *testing.T) {
2855 testCases := []struct {
2856 name string
2857 bp string
2858 noCode bool
2859 }{
2860 {
2861 name: "normal",
2862 bp: `
2863 android_app {
2864 name: "foo",
2865 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002866 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002867 }
2868 `,
2869 noCode: false,
2870 },
2871 {
2872 name: "app without sources",
2873 bp: `
2874 android_app {
2875 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002876 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002877 }
2878 `,
2879 noCode: true,
2880 },
2881 {
2882 name: "app with libraries",
2883 bp: `
2884 android_app {
2885 name: "foo",
2886 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002887 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002888 }
2889
2890 java_library {
2891 name: "lib",
2892 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002893 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002894 }
2895 `,
2896 noCode: false,
2897 },
2898 {
2899 name: "app with sourceless libraries",
2900 bp: `
2901 android_app {
2902 name: "foo",
2903 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002904 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002905 }
2906
2907 java_library {
2908 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002909 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002910 }
2911 `,
2912 // TODO(jungjw): this should probably be true
2913 noCode: false,
2914 },
2915 }
2916
2917 for _, test := range testCases {
2918 t.Run(test.name, func(t *testing.T) {
2919 ctx := testApp(t, test.bp)
2920
2921 foo := ctx.ModuleForTests("foo", "android_common")
2922 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2923 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2924 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2925 }
2926 })
2927 }
2928}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002929
2930func TestEmbedNotice(t *testing.T) {
Colin Cross238c1f32020-06-07 16:58:18 -07002931 ctx, _ := testJavaWithFS(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002932 android_app {
2933 name: "foo",
2934 srcs: ["a.java"],
2935 static_libs: ["javalib"],
2936 jni_libs: ["libjni"],
2937 notice: "APP_NOTICE",
2938 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002939 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002940 }
2941
2942 // No embed_notice flag
2943 android_app {
2944 name: "bar",
2945 srcs: ["a.java"],
2946 jni_libs: ["libjni"],
2947 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002948 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002949 }
2950
2951 // No NOTICE files
2952 android_app {
2953 name: "baz",
2954 srcs: ["a.java"],
2955 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002956 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002957 }
2958
2959 cc_library {
2960 name: "libjni",
2961 system_shared_libs: [],
2962 stl: "none",
2963 notice: "LIB_NOTICE",
Colin Cross094cde42020-02-15 10:38:00 -08002964 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002965 }
2966
2967 java_library {
2968 name: "javalib",
2969 srcs: [
2970 ":gen",
2971 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002972 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002973 }
2974
2975 genrule {
2976 name: "gen",
2977 tools: ["gentool"],
2978 out: ["gen.java"],
2979 notice: "GENRULE_NOTICE",
2980 }
2981
2982 java_binary_host {
2983 name: "gentool",
2984 srcs: ["b.java"],
2985 notice: "TOOL_NOTICE",
2986 }
Colin Cross238c1f32020-06-07 16:58:18 -07002987 `, map[string][]byte{
2988 "APP_NOTICE": nil,
2989 "GENRULE_NOTICE": nil,
2990 "LIB_NOTICE": nil,
2991 "TOOL_NOTICE": nil,
2992 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002993
2994 // foo has NOTICE files to process, and embed_notices is true.
2995 foo := ctx.ModuleForTests("foo", "android_common")
2996 // verify merge notices rule.
2997 mergeNotices := foo.Rule("mergeNoticesRule")
2998 noticeInputs := mergeNotices.Inputs.Strings()
2999 // TOOL_NOTICE should be excluded as it's a host module.
3000 if len(mergeNotices.Inputs) != 3 {
3001 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
3002 }
3003 if !inList("APP_NOTICE", noticeInputs) {
3004 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
3005 }
3006 if !inList("LIB_NOTICE", noticeInputs) {
3007 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
3008 }
3009 if !inList("GENRULE_NOTICE", noticeInputs) {
3010 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
3011 }
3012 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
3013 res := foo.Output("package-res.apk")
3014 aapt2Flags := res.Args["flags"]
3015 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
3016 if !strings.Contains(aapt2Flags, e) {
3017 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
3018 }
3019
3020 // bar has NOTICE files to process, but embed_notices is not set.
3021 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07003022 res = bar.Output("package-res.apk")
3023 aapt2Flags = res.Args["flags"]
3024 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
3025 if strings.Contains(aapt2Flags, e) {
3026 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07003027 }
3028
3029 // baz's embed_notice is true, but it doesn't have any NOTICE files.
3030 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07003031 res = baz.Output("package-res.apk")
3032 aapt2Flags = res.Args["flags"]
3033 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
3034 if strings.Contains(aapt2Flags, e) {
3035 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07003036 }
3037}
Colin Cross53a87f52019-06-25 13:35:30 -07003038
3039func TestUncompressDex(t *testing.T) {
3040 testCases := []struct {
3041 name string
3042 bp string
3043
3044 uncompressedPlatform bool
3045 uncompressedUnbundled bool
3046 }{
3047 {
3048 name: "normal",
3049 bp: `
3050 android_app {
3051 name: "foo",
3052 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09003053 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07003054 }
3055 `,
3056 uncompressedPlatform: true,
3057 uncompressedUnbundled: false,
3058 },
3059 {
3060 name: "use_embedded_dex",
3061 bp: `
3062 android_app {
3063 name: "foo",
3064 use_embedded_dex: true,
3065 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09003066 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07003067 }
3068 `,
3069 uncompressedPlatform: true,
3070 uncompressedUnbundled: true,
3071 },
3072 {
3073 name: "privileged",
3074 bp: `
3075 android_app {
3076 name: "foo",
3077 privileged: true,
3078 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09003079 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07003080 }
3081 `,
3082 uncompressedPlatform: true,
3083 uncompressedUnbundled: true,
3084 },
David Srbeckye033cba2020-05-20 22:20:28 +01003085 {
3086 name: "normal_uncompress_dex_true",
3087 bp: `
3088 android_app {
3089 name: "foo",
3090 srcs: ["a.java"],
3091 sdk_version: "current",
3092 uncompress_dex: true,
3093 }
3094 `,
3095 uncompressedPlatform: true,
3096 uncompressedUnbundled: true,
3097 },
3098 {
3099 name: "normal_uncompress_dex_false",
3100 bp: `
3101 android_app {
3102 name: "foo",
3103 srcs: ["a.java"],
3104 sdk_version: "current",
3105 uncompress_dex: false,
3106 }
3107 `,
3108 uncompressedPlatform: false,
3109 uncompressedUnbundled: false,
3110 },
Colin Cross53a87f52019-06-25 13:35:30 -07003111 }
3112
3113 test := func(t *testing.T, bp string, want bool, unbundled bool) {
3114 t.Helper()
3115
Colin Cross98be1bb2019-12-13 20:41:13 -08003116 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07003117 if unbundled {
3118 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
Jeongik Cha816a23a2020-07-08 01:09:23 +09003119 config.TestProductVariables.Always_use_prebuilt_sdks = proptools.BoolPtr(true)
Colin Cross53a87f52019-06-25 13:35:30 -07003120 }
3121
Colin Cross98be1bb2019-12-13 20:41:13 -08003122 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07003123
3124 run(t, ctx, config)
3125
3126 foo := ctx.ModuleForTests("foo", "android_common")
3127 dex := foo.Rule("r8")
3128 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
3129 aligned := foo.MaybeRule("zipalign").Rule != nil
3130
3131 if uncompressedInDexJar != want {
3132 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
3133 }
3134
3135 if aligned != want {
3136 t.Errorf("want aligned %v, got %v", want, aligned)
3137 }
3138 }
3139
3140 for _, tt := range testCases {
3141 t.Run(tt.name, func(t *testing.T) {
3142 t.Run("platform", func(t *testing.T) {
3143 test(t, tt.bp, tt.uncompressedPlatform, false)
3144 })
3145 t.Run("unbundled", func(t *testing.T) {
3146 test(t, tt.bp, tt.uncompressedUnbundled, true)
3147 })
3148 })
3149 }
3150}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07003151
3152func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
3153 if expectedValue != "" {
3154 expectedFlag := "--" + flagName + " " + expectedValue
3155 if !strings.Contains(aapt2Flags, expectedFlag) {
3156 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
3157 }
3158 } else {
3159 unexpectedFlag := "--" + flagName
3160 if strings.Contains(aapt2Flags, unexpectedFlag) {
3161 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
3162 }
3163 }
3164}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003165
3166func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003167 fs := map[string][]byte{
3168 "baz/res/res/values/strings.xml": nil,
3169 "bar/res/res/values/strings.xml": nil,
3170 }
3171 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003172 runtime_resource_overlay {
3173 name: "foo",
3174 certificate: "platform",
Liz Kammer966b2f02020-05-19 16:15:25 -07003175 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003176 product_specific: true,
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003177 static_libs: ["bar"],
3178 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003179 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003180 }
3181
3182 runtime_resource_overlay {
3183 name: "foo_themed",
3184 certificate: "platform",
3185 product_specific: true,
3186 theme: "faza",
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003187 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003188 }
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003189
3190 android_library {
3191 name: "bar",
3192 resource_dirs: ["bar/res"],
3193 }
3194
3195 android_app {
3196 name: "baz",
3197 sdk_version: "current",
3198 resource_dirs: ["baz/res"],
3199 }
3200 `
3201 config := testAppConfig(nil, bp, fs)
3202 ctx := testContext()
3203 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003204
3205 m := ctx.ModuleForTests("foo", "android_common")
3206
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003207 // Check AAPT2 link flags.
3208 aapt2Flags := m.Output("package-res.apk").Args["flags"]
3209 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3210 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
3211 if len(absentFlags) > 0 {
3212 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3213 }
3214
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003215 // Check overlay.list output for static_libs dependency.
3216 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
3217 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
3218 if !inList(staticLibPackage, overlayList) {
3219 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
3220 }
3221
3222 // Check AAPT2 link flags for resource_libs dependency.
3223 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
3224 if !strings.Contains(aapt2Flags, resourceLibFlag) {
3225 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
3226 }
3227
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003228 // Check cert signing flag.
3229 signedApk := m.Output("signed/foo.apk")
Liz Kammer966b2f02020-05-19 16:15:25 -07003230 lineageFlag := signedApk.Args["flags"]
3231 expectedLineageFlag := "--lineage lineage.bin"
3232 if expectedLineageFlag != lineageFlag {
3233 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
3234 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003235 signingFlag := signedApk.Args["certificates"]
3236 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
3237 if expected != signingFlag {
3238 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
3239 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003240 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3241 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003242 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
3243 if !reflect.DeepEqual(path, expectedPath) {
3244 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
3245 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003246
3247 // Check device location.
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003248 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003249 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003250 if !reflect.DeepEqual(path, expectedPath) {
3251 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3252 }
3253
3254 // A themed module has a different device location
3255 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003256 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3257 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003258 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3259 if !reflect.DeepEqual(path, expectedPath) {
3260 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3261 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003262
3263 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3264 expectedOverrides := []string{"foo"}
3265 if !reflect.DeepEqual(overrides, expectedOverrides) {
3266 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3267 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003268}
Roshan Pius4df2bc72020-04-27 09:42:27 -07003269
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07003270func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3271 ctx, config := testJava(t, `
3272 java_defaults {
3273 name: "rro_defaults",
3274 theme: "default_theme",
3275 product_specific: true,
3276 aaptflags: ["--keep-raw-values"],
3277 }
3278
3279 runtime_resource_overlay {
3280 name: "foo_with_defaults",
3281 defaults: ["rro_defaults"],
3282 }
3283
3284 runtime_resource_overlay {
3285 name: "foo_barebones",
3286 }
3287 `)
3288
3289 //
3290 // RRO module with defaults
3291 //
3292 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3293
3294 // Check AAPT2 link flags.
3295 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3296 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3297 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3298 if len(absentFlags) > 0 {
3299 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3300 }
3301
3302 // Check device location.
3303 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3304 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3305 if !reflect.DeepEqual(path, expectedPath) {
3306 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3307 }
3308
3309 //
3310 // RRO module without defaults
3311 //
3312 m = ctx.ModuleForTests("foo_barebones", "android_common")
3313
3314 // Check AAPT2 link flags.
3315 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3316 unexpectedFlags := "--keep-raw-values"
3317 if inList(unexpectedFlags, aapt2Flags) {
3318 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3319 }
3320
3321 // Check device location.
3322 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3323 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3324 if !reflect.DeepEqual(path, expectedPath) {
3325 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3326 }
3327}
Roshan Piusb8307962020-04-27 09:42:27 -07003328
Roshan Pius4df2bc72020-04-27 09:42:27 -07003329func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3330 ctx, _ := testJava(t, `
3331 runtime_resource_overlay {
3332 name: "foo_overlay",
3333 certificate: "platform",
3334 product_specific: true,
3335 sdk_version: "current",
3336 }
3337
3338 override_runtime_resource_overlay {
3339 name: "bar_overlay",
3340 base: "foo_overlay",
3341 package_name: "com.android.bar.overlay",
3342 target_package_name: "com.android.bar",
3343 }
3344 `)
3345
3346 expectedVariants := []struct {
3347 moduleName string
3348 variantName string
3349 apkPath string
3350 overrides []string
3351 targetVariant string
3352 packageFlag string
3353 targetPackageFlag string
3354 }{
3355 {
3356 variantName: "android_common",
3357 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3358 overrides: nil,
3359 targetVariant: "android_common",
3360 packageFlag: "",
3361 targetPackageFlag: "",
3362 },
3363 {
3364 variantName: "android_common_bar_overlay",
3365 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3366 overrides: []string{"foo_overlay"},
3367 targetVariant: "android_common_bar",
3368 packageFlag: "com.android.bar.overlay",
3369 targetPackageFlag: "com.android.bar",
3370 },
3371 }
3372 for _, expected := range expectedVariants {
3373 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3374
3375 // Check the final apk name
3376 outputs := variant.AllOutputs()
3377 expectedApkPath := buildDir + expected.apkPath
3378 found := false
3379 for _, o := range outputs {
3380 if o == expectedApkPath {
3381 found = true
3382 break
3383 }
3384 }
3385 if !found {
3386 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3387 }
3388
3389 // Check if the overrides field values are correctly aggregated.
3390 mod := variant.Module().(*RuntimeResourceOverlay)
3391 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3392 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3393 expected.overrides, mod.properties.Overrides)
3394 }
3395
3396 // Check aapt2 flags.
3397 res := variant.Output("package-res.apk")
3398 aapt2Flags := res.Args["flags"]
3399 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammer9f9fd022020-06-18 19:44:06 +00003400 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", "")
Roshan Pius4df2bc72020-04-27 09:42:27 -07003401 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3402 }
3403}