blob: b8d8616f36b0c893dbd0df84c55866f0d6e0f53b [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 {
179 name string
180 deviceArch *string
181 deviceSecondaryArch *string
182 aaptPrebuiltDPI []string
183 sdkVersion int
184 expected map[string]string
185 }{
186 {
187 name: "One",
188 deviceArch: proptools.StringPtr("x86"),
189 aaptPrebuiltDPI: []string{"ldpi", "xxhdpi"},
190 sdkVersion: 29,
191 expected: map[string]string{
192 "abis": "X86",
193 "allow-prereleased": "false",
194 "screen-densities": "LDPI,XXHDPI",
195 "sdk-version": "29",
196 "stem": "foo",
197 },
198 },
199 {
200 name: "Two",
201 deviceArch: proptools.StringPtr("x86_64"),
202 deviceSecondaryArch: proptools.StringPtr("x86"),
203 aaptPrebuiltDPI: nil,
204 sdkVersion: 30,
205 expected: map[string]string{
206 "abis": "X86_64,X86",
207 "allow-prereleased": "false",
208 "screen-densities": "all",
209 "sdk-version": "30",
210 "stem": "foo",
211 },
212 },
213 }
214
215 for _, test := range testCases {
216 config := testAppConfig(nil, bp, nil)
217 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
218 config.TestProductVariables.Platform_sdk_version = &test.sdkVersion
219 config.TestProductVariables.DeviceArch = test.deviceArch
220 config.TestProductVariables.DeviceSecondaryArch = test.deviceSecondaryArch
221 ctx := testContext()
222 run(t, ctx, config)
223 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundak18d98bc2020-05-27 16:36:07 -0700224 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-04-23 09:49:59 -0700225 params := module.Output(packedSplitApks)
226 for k, v := range test.expected {
227 if actual := params.Args[k]; actual != v {
228 t.Errorf("%s: bad build arg value for '%s': '%s', expected '%s'",
229 test.name, k, actual, v)
230 }
231 }
232 }
233}
234
Jeongik Cha538c0d02019-07-11 15:54:27 +0900235func TestPlatformAPIs(t *testing.T) {
236 testJava(t, `
237 android_app {
238 name: "foo",
239 srcs: ["a.java"],
240 platform_apis: true,
241 }
242 `)
243
244 testJava(t, `
245 android_app {
246 name: "foo",
247 srcs: ["a.java"],
248 sdk_version: "current",
249 }
250 `)
251
252 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
253 android_app {
254 name: "bar",
255 srcs: ["b.java"],
256 }
257 `)
258
259 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
260 android_app {
261 name: "bar",
262 srcs: ["b.java"],
263 sdk_version: "system_current",
264 platform_apis: true,
265 }
266 `)
267}
268
Jeongik Chae403e9e2019-12-07 00:16:24 +0900269func TestAndroidAppLinkType(t *testing.T) {
270 testJava(t, `
271 android_app {
272 name: "foo",
273 srcs: ["a.java"],
274 libs: ["bar"],
275 static_libs: ["baz"],
276 platform_apis: true,
277 }
278
279 java_library {
280 name: "bar",
281 sdk_version: "current",
282 srcs: ["b.java"],
283 }
284
285 android_library {
286 name: "baz",
287 sdk_version: "system_current",
288 srcs: ["c.java"],
289 }
290 `)
291
292 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
293 android_app {
294 name: "foo",
295 srcs: ["a.java"],
296 libs: ["bar"],
297 sdk_version: "current",
298 static_libs: ["baz"],
299 }
300
301 java_library {
302 name: "bar",
303 sdk_version: "current",
304 srcs: ["b.java"],
305 }
306
307 android_library {
308 name: "baz",
309 sdk_version: "system_current",
310 srcs: ["c.java"],
311 }
312 `)
313
314 testJava(t, `
315 android_app {
316 name: "foo",
317 srcs: ["a.java"],
318 libs: ["bar"],
319 sdk_version: "system_current",
320 static_libs: ["baz"],
321 }
322
323 java_library {
324 name: "bar",
325 sdk_version: "current",
326 srcs: ["b.java"],
327 }
328
329 android_library {
330 name: "baz",
331 sdk_version: "system_current",
332 srcs: ["c.java"],
333 }
334 `)
335
336 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
337 android_app {
338 name: "foo",
339 srcs: ["a.java"],
340 libs: ["bar"],
341 sdk_version: "system_current",
342 static_libs: ["baz"],
343 }
344
345 java_library {
346 name: "bar",
347 sdk_version: "current",
348 srcs: ["b.java"],
349 }
350
351 android_library {
352 name: "baz",
353 srcs: ["c.java"],
354 }
355 `)
356}
357
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100358func TestUpdatableApps(t *testing.T) {
359 testCases := []struct {
360 name string
361 bp string
362 expectedError string
363 }{
364 {
365 name: "Stable public SDK",
366 bp: `android_app {
367 name: "foo",
368 srcs: ["a.java"],
369 sdk_version: "29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100370 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100371 updatable: true,
372 }`,
373 },
374 {
375 name: "Stable system SDK",
376 bp: `android_app {
377 name: "foo",
378 srcs: ["a.java"],
379 sdk_version: "system_29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100380 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100381 updatable: true,
382 }`,
383 },
384 {
385 name: "Current public SDK",
386 bp: `android_app {
387 name: "foo",
388 srcs: ["a.java"],
389 sdk_version: "current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100390 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100391 updatable: true,
392 }`,
393 },
394 {
395 name: "Current system SDK",
396 bp: `android_app {
397 name: "foo",
398 srcs: ["a.java"],
399 sdk_version: "system_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100400 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100401 updatable: true,
402 }`,
403 },
404 {
405 name: "Current module SDK",
406 bp: `android_app {
407 name: "foo",
408 srcs: ["a.java"],
409 sdk_version: "module_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100410 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100411 updatable: true,
412 }`,
413 },
414 {
415 name: "Current core SDK",
416 bp: `android_app {
417 name: "foo",
418 srcs: ["a.java"],
419 sdk_version: "core_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100420 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100421 updatable: true,
422 }`,
423 },
424 {
425 name: "No Platform APIs",
426 bp: `android_app {
427 name: "foo",
428 srcs: ["a.java"],
429 platform_apis: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100430 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100431 updatable: true,
432 }`,
433 expectedError: "Updatable apps must use stable SDKs",
434 },
435 {
436 name: "No Core Platform APIs",
437 bp: `android_app {
438 name: "foo",
439 srcs: ["a.java"],
440 sdk_version: "core_platform",
Artur Satayevf40fc852020-04-16 13:43:02 +0100441 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100442 updatable: true,
443 }`,
444 expectedError: "Updatable apps must use stable SDKs",
445 },
446 {
447 name: "No unspecified APIs",
448 bp: `android_app {
449 name: "foo",
450 srcs: ["a.java"],
451 updatable: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100452 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100453 }`,
454 expectedError: "Updatable apps must use stable SDK",
455 },
Artur Satayevf40fc852020-04-16 13:43:02 +0100456 {
457 name: "Must specify min_sdk_version",
458 bp: `android_app {
459 name: "app_without_min_sdk_version",
460 srcs: ["a.java"],
461 sdk_version: "29",
462 updatable: true,
463 }`,
464 expectedError: "updatable apps must set min_sdk_version.",
465 },
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100466 }
467
468 for _, test := range testCases {
469 t.Run(test.name, func(t *testing.T) {
470 if test.expectedError == "" {
471 testJava(t, test.bp)
472 } else {
473 testJavaError(t, test.expectedError, test.bp)
474 }
475 })
476 }
477}
478
Jooyung Han749dc692020-04-15 11:03:39 +0900479func TestUpdatableApps_TransitiveDepsShouldSetMinSdkVersion(t *testing.T) {
480 testJavaError(t, `module "bar".*: should support min_sdk_version\(29\)`, cc.GatherRequiredDepsForTest(android.Android)+`
481 android_app {
482 name: "foo",
483 srcs: ["a.java"],
484 updatable: true,
485 sdk_version: "current",
486 min_sdk_version: "29",
487 static_libs: ["bar"],
488 }
489
490 java_library {
491 name: "bar",
492 sdk_version: "current",
493 }
494 `)
495}
496
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900497func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
498 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
499 android_app {
500 name: "foo",
501 srcs: ["a.java"],
502 updatable: true,
503 sdk_version: "current",
504 min_sdk_version: "current",
505 jni_libs: ["libjni"],
506 }
507
508 cc_library {
509 name: "libjni",
510 stl: "none",
511 system_shared_libs: [],
512 sdk_version: "current",
513 }
514 `)
515}
516
517func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
518 bp := cc.GatherRequiredDepsForTest(android.Android) + `
519 android_app {
520 name: "foo",
521 srcs: ["a.java"],
522 updatable: true,
523 sdk_version: "current",
524 min_sdk_version: "29",
525 jni_libs: ["libjni"],
526 }
527
528 cc_library {
529 name: "libjni",
530 stl: "none",
531 system_shared_libs: [],
532 sdk_version: "29",
533 }
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900534 `
535 fs := map[string][]byte{
536 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
537 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
538 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
539 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
540 }
541
542 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
543
544 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
545 var crtbeginFound, crtendFound bool
Dan Albert92fe7402020-07-15 13:33:30 -0700546 expectedCrtBegin := ctx.ModuleForTests("crtbegin_so",
547 "android_arm64_armv8-a_sdk_29").Rule("partialLd").Output
548 expectedCrtEnd := ctx.ModuleForTests("crtend_so",
549 "android_arm64_armv8-a_sdk_29").Rule("partialLd").Output
550 implicits := []string{}
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900551 for _, input := range inputs {
Dan Albert92fe7402020-07-15 13:33:30 -0700552 implicits = append(implicits, input.String())
553 if strings.HasSuffix(input.String(), expectedCrtBegin.String()) {
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900554 crtbeginFound = true
Dan Albert92fe7402020-07-15 13:33:30 -0700555 } else if strings.HasSuffix(input.String(), expectedCrtEnd.String()) {
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900556 crtendFound = true
557 }
558 }
Dan Albert92fe7402020-07-15 13:33:30 -0700559 if !crtbeginFound {
560 t.Error(fmt.Sprintf(
561 "expected implicit with suffix %q, have the following implicits:\n%s",
562 expectedCrtBegin, strings.Join(implicits, "\n")))
563 }
564 if !crtendFound {
565 t.Error(fmt.Sprintf(
566 "expected implicit with suffix %q, have the following implicits:\n%s",
567 expectedCrtEnd, strings.Join(implicits, "\n")))
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900568 }
569}
570
571func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
572 bp := cc.GatherRequiredDepsForTest(android.Android) + `
573 android_app {
574 name: "foo",
575 srcs: ["a.java"],
576 updatable: true,
577 sdk_version: "current",
578 min_sdk_version: "29", // this APK should support 29
579 jni_libs: ["libjni"],
580 }
581
582 cc_library {
583 name: "libjni",
584 stl: "none",
585 sdk_version: "current",
586 }
587 `
588 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
589}
590
591func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
592 bp := cc.GatherRequiredDepsForTest(android.Android) + `
593 android_app {
594 name: "foo",
595 srcs: ["a.java"],
596 updatable: true,
597 sdk_version: "current",
598 min_sdk_version: "29", // this APK should support 29
599 jni_libs: ["libjni"],
600 }
601
602 cc_library {
603 name: "libjni",
604 stl: "none",
605 shared_libs: ["libbar"],
606 system_shared_libs: [],
607 sdk_version: "27",
608 }
609
610 cc_library {
611 name: "libbar",
612 stl: "none",
613 system_shared_libs: [],
614 sdk_version: "current",
615 }
616 `
617 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
618}
619
Colin Cross0ddae7f2019-02-07 15:30:01 -0800620func TestResourceDirs(t *testing.T) {
621 testCases := []struct {
622 name string
623 prop string
624 resources []string
625 }{
626 {
627 name: "no resource_dirs",
628 prop: "",
629 resources: []string{"res/res/values/strings.xml"},
630 },
631 {
632 name: "resource_dirs",
633 prop: `resource_dirs: ["res"]`,
634 resources: []string{"res/res/values/strings.xml"},
635 },
636 {
637 name: "empty resource_dirs",
638 prop: `resource_dirs: []`,
639 resources: nil,
640 },
641 }
642
643 fs := map[string][]byte{
644 "res/res/values/strings.xml": nil,
645 }
646
647 bp := `
648 android_app {
649 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900650 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800651 %s
652 }
653 `
654
655 for _, testCase := range testCases {
656 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800657 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
658 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800659 run(t, ctx, config)
660
661 module := ctx.ModuleForTests("foo", "android_common")
662 resourceList := module.MaybeOutput("aapt2/res.list")
663
664 var resources []string
665 if resourceList.Rule != nil {
666 for _, compiledResource := range resourceList.Inputs.Strings() {
667 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
668 }
669 }
670
671 if !reflect.DeepEqual(resources, testCase.resources) {
672 t.Errorf("expected resource files %q, got %q",
673 testCase.resources, resources)
674 }
675 })
676 }
677}
678
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800679func TestLibraryAssets(t *testing.T) {
680 bp := `
681 android_app {
682 name: "foo",
683 sdk_version: "current",
684 static_libs: ["lib1", "lib2", "lib3"],
685 }
686
687 android_library {
688 name: "lib1",
689 sdk_version: "current",
690 asset_dirs: ["assets_a"],
691 }
692
693 android_library {
694 name: "lib2",
695 sdk_version: "current",
696 }
697
698 android_library {
699 name: "lib3",
700 sdk_version: "current",
701 static_libs: ["lib4"],
702 }
703
704 android_library {
705 name: "lib4",
706 sdk_version: "current",
707 asset_dirs: ["assets_b"],
708 }
709 `
710
711 testCases := []struct {
712 name string
713 assetFlag string
714 assetPackages []string
715 }{
716 {
717 name: "foo",
718 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
719 assetPackages: []string{
720 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
721 buildDir + "/.intermediates/lib1/android_common/assets.zip",
722 buildDir + "/.intermediates/lib3/android_common/assets.zip",
723 },
724 },
725 {
726 name: "lib1",
727 assetFlag: "-A assets_a",
728 },
729 {
730 name: "lib2",
731 },
732 {
733 name: "lib3",
734 assetPackages: []string{
735 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
736 buildDir + "/.intermediates/lib4/android_common/assets.zip",
737 },
738 },
739 {
740 name: "lib4",
741 assetFlag: "-A assets_b",
742 },
743 }
744 ctx := testApp(t, bp)
745
746 for _, test := range testCases {
747 t.Run(test.name, func(t *testing.T) {
748 m := ctx.ModuleForTests(test.name, "android_common")
749
750 // Check asset flag in aapt2 link flags
751 var aapt2link android.TestingBuildParams
752 if len(test.assetPackages) > 0 {
753 aapt2link = m.Output("aapt2/package-res.apk")
754 } else {
755 aapt2link = m.Output("package-res.apk")
756 }
757 aapt2Flags := aapt2link.Args["flags"]
758 if test.assetFlag != "" {
759 if !strings.Contains(aapt2Flags, test.assetFlag) {
760 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
761 }
762 } else {
763 if strings.Contains(aapt2Flags, " -A ") {
764 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
765 }
766 }
767
768 // Check asset merge rule.
769 if len(test.assetPackages) > 0 {
770 mergeAssets := m.Output("package-res.apk")
771 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
772 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
773 mergeAssets.Inputs.Strings(), test.assetPackages)
774 }
775 }
776 })
777 }
778}
779
Colin Crossbec85302019-02-13 13:15:46 -0800780func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800781 testCases := []struct {
782 name string
783 enforceRROTargets []string
784 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800785 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800786 overlayFiles map[string][]string
787 rroDirs map[string][]string
788 }{
789 {
790 name: "no RRO",
791 enforceRROTargets: nil,
792 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800793 resourceFiles: map[string][]string{
794 "foo": nil,
795 "bar": {"bar/res/res/values/strings.xml"},
796 "lib": nil,
797 "lib2": {"lib2/res/res/values/strings.xml"},
798 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800799 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800800 "foo": {
801 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800802 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000803 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800804 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800805 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
806 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000807 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800808 },
Colin Crossbec85302019-02-13 13:15:46 -0800809 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800810 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
811 "device/vendor/blah/overlay/bar/res/values/strings.xml",
812 },
Colin Crossbec85302019-02-13 13:15:46 -0800813 "lib": {
814 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
815 "lib/res/res/values/strings.xml",
816 "device/vendor/blah/overlay/lib/res/values/strings.xml",
817 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800818 },
819 rroDirs: map[string][]string{
820 "foo": nil,
821 "bar": nil,
822 },
823 },
824 {
825 name: "enforce RRO on foo",
826 enforceRROTargets: []string{"foo"},
827 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800828 resourceFiles: map[string][]string{
829 "foo": nil,
830 "bar": {"bar/res/res/values/strings.xml"},
831 "lib": nil,
832 "lib2": {"lib2/res/res/values/strings.xml"},
833 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800834 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800835 "foo": {
836 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800837 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000838 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800839 "foo/res/res/values/strings.xml",
840 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
841 },
Colin Crossbec85302019-02-13 13:15:46 -0800842 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800843 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
844 "device/vendor/blah/overlay/bar/res/values/strings.xml",
845 },
Colin Crossbec85302019-02-13 13:15:46 -0800846 "lib": {
847 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
848 "lib/res/res/values/strings.xml",
849 "device/vendor/blah/overlay/lib/res/values/strings.xml",
850 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800851 },
Colin Crossc1c37552019-01-31 11:42:41 -0800852
Colin Cross5c4791c2019-02-01 11:44:44 -0800853 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800854 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000855 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800856 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
857 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000858 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800859 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800860 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800861 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800862 },
863 },
864 {
865 name: "enforce RRO on all",
866 enforceRROTargets: []string{"*"},
867 enforceRROExcludedOverlays: []string{
868 // Excluding specific apps/res directories also allowed.
869 "device/vendor/blah/static_overlay/foo",
870 "device/vendor/blah/static_overlay/bar/res",
871 },
Colin Crossbec85302019-02-13 13:15:46 -0800872 resourceFiles: map[string][]string{
873 "foo": nil,
874 "bar": {"bar/res/res/values/strings.xml"},
875 "lib": nil,
876 "lib2": {"lib2/res/res/values/strings.xml"},
877 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800878 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800879 "foo": {
880 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800881 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000882 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800883 "foo/res/res/values/strings.xml",
884 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
885 },
Colin Crossbec85302019-02-13 13:15:46 -0800886 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
887 "lib": {
888 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
889 "lib/res/res/values/strings.xml",
890 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800891 },
892 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800893 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000894 "device:device/vendor/blah/overlay/foo/res",
895 "product:product/vendor/blah/overlay/foo/res",
896 // Lib dep comes after the direct deps
897 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800898 },
Anton Hansson53c88442019-03-18 15:53:16 +0000899 "bar": {"device:device/vendor/blah/overlay/bar/res"},
900 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800901 },
902 },
903 }
904
Anton Hansson53c88442019-03-18 15:53:16 +0000905 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800906 "device/vendor/blah/overlay",
907 "device/vendor/blah/overlay2",
908 "device/vendor/blah/static_overlay",
909 }
910
Anton Hansson53c88442019-03-18 15:53:16 +0000911 productResourceOverlays := []string{
912 "product/vendor/blah/overlay",
913 }
914
Colin Cross890ff552017-11-30 20:13:19 -0800915 fs := map[string][]byte{
916 "foo/res/res/values/strings.xml": nil,
917 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800918 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800919 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800920 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
921 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800922 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800923 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
924 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
925 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000926 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800927 }
928
929 bp := `
930 android_app {
931 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900932 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800933 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000934 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800935 }
936
937 android_app {
938 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900939 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800940 resource_dirs: ["bar/res"],
941 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800942
943 android_library {
944 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900945 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800946 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800947 static_libs: ["lib2"],
948 }
949
950 android_library {
951 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900952 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800953 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800954 }
Anton Hansson53c88442019-03-18 15:53:16 +0000955
956 // This library has the same resources as lib (should not lead to dupe RROs)
957 android_library {
958 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900959 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000960 resource_dirs: ["lib/res"]
961 }
Colin Cross890ff552017-11-30 20:13:19 -0800962 `
963
Colin Cross5c4791c2019-02-01 11:44:44 -0800964 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800965 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800966 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000967 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
968 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800969 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800970 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800971 }
972 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800973 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800974 }
975
Colin Cross98be1bb2019-12-13 20:41:13 -0800976 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800977 run(t, ctx, config)
978
Colin Crossbec85302019-02-13 13:15:46 -0800979 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
980 for _, o := range list {
981 res := module.MaybeOutput(o)
982 if res.Rule != nil {
983 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
984 // verify the inputs to the .arsc.flat rule.
985 files = append(files, res.Inputs.Strings()...)
986 } else {
987 // Otherwise, verify the full path to the output of the other module
988 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000989 }
Colin Cross890ff552017-11-30 20:13:19 -0800990 }
Colin Crossbec85302019-02-13 13:15:46 -0800991 return files
Colin Cross890ff552017-11-30 20:13:19 -0800992 }
993
Colin Crossbec85302019-02-13 13:15:46 -0800994 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
995 module := ctx.ModuleForTests(moduleName, "android_common")
996 resourceList := module.MaybeOutput("aapt2/res.list")
997 if resourceList.Rule != nil {
998 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000999 }
Colin Crossbec85302019-02-13 13:15:46 -08001000 overlayList := module.MaybeOutput("aapt2/overlay.list")
1001 if overlayList.Rule != nil {
1002 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
1003 }
1004
Anton Hansson53c88442019-03-18 15:53:16 +00001005 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
1006 var prefix string
1007 if d.overlayType == device {
1008 prefix = "device:"
1009 } else if d.overlayType == product {
1010 prefix = "product:"
1011 } else {
1012 t.Fatalf("Unexpected overlayType %d", d.overlayType)
1013 }
1014 rroDirs = append(rroDirs, prefix+d.path.String())
1015 }
Colin Crossbec85302019-02-13 13:15:46 -08001016
1017 return resourceFiles, overlayFiles, rroDirs
1018 }
1019
1020 modules := []string{"foo", "bar", "lib", "lib2"}
1021 for _, module := range modules {
1022 resourceFiles, overlayFiles, rroDirs := getResources(module)
1023
1024 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
1025 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
1026 module, testCase.resourceFiles[module], resourceFiles)
1027 }
1028 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
1029 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
1030 module, testCase.overlayFiles[module], overlayFiles)
1031 }
1032 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +00001033 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -08001034 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +00001035 }
Colin Cross890ff552017-11-30 20:13:19 -08001036 }
Colin Cross890ff552017-11-30 20:13:19 -08001037 })
1038 }
1039}
Colin Crossd09b0b62018-04-18 11:06:47 -07001040
Jeongik Cha219141c2020-08-06 23:00:37 +09001041func checkSdkVersion(t *testing.T, config android.Config, expectedSdkVersion string) {
1042 ctx := testContext()
1043
1044 run(t, ctx, config)
1045
1046 foo := ctx.ModuleForTests("foo", "android_common")
1047 link := foo.Output("package-res.apk")
1048 linkFlags := strings.Split(link.Args["flags"], " ")
1049 min := android.IndexList("--min-sdk-version", linkFlags)
1050 target := android.IndexList("--target-sdk-version", linkFlags)
1051
1052 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
1053 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
1054 }
1055
1056 gotMinSdkVersion := linkFlags[min+1]
1057 gotTargetSdkVersion := linkFlags[target+1]
1058
1059 if gotMinSdkVersion != expectedSdkVersion {
1060 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
1061 expectedSdkVersion, gotMinSdkVersion)
1062 }
1063
1064 if gotTargetSdkVersion != expectedSdkVersion {
1065 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
1066 expectedSdkVersion, gotTargetSdkVersion)
1067 }
1068}
1069
Colin Crossd09b0b62018-04-18 11:06:47 -07001070func TestAppSdkVersion(t *testing.T) {
1071 testCases := []struct {
1072 name string
1073 sdkVersion string
1074 platformSdkInt int
1075 platformSdkCodename string
1076 platformSdkFinal bool
1077 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +09001078 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -07001079 }{
1080 {
1081 name: "current final SDK",
1082 sdkVersion: "current",
1083 platformSdkInt: 27,
1084 platformSdkCodename: "REL",
1085 platformSdkFinal: true,
1086 expectedMinSdkVersion: "27",
1087 },
1088 {
1089 name: "current non-final SDK",
1090 sdkVersion: "current",
1091 platformSdkInt: 27,
1092 platformSdkCodename: "OMR1",
1093 platformSdkFinal: false,
1094 expectedMinSdkVersion: "OMR1",
1095 },
1096 {
1097 name: "default final SDK",
1098 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001099 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001100 platformSdkInt: 27,
1101 platformSdkCodename: "REL",
1102 platformSdkFinal: true,
1103 expectedMinSdkVersion: "27",
1104 },
1105 {
1106 name: "default non-final SDK",
1107 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001108 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001109 platformSdkInt: 27,
1110 platformSdkCodename: "OMR1",
1111 platformSdkFinal: false,
1112 expectedMinSdkVersion: "OMR1",
1113 },
1114 {
1115 name: "14",
1116 sdkVersion: "14",
1117 expectedMinSdkVersion: "14",
1118 },
1119 }
1120
1121 for _, moduleType := range []string{"android_app", "android_library"} {
1122 for _, test := range testCases {
1123 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001124 platformApiProp := ""
1125 if test.platformApis {
1126 platformApiProp = "platform_apis: true,"
1127 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001128 bp := fmt.Sprintf(`%s {
1129 name: "foo",
1130 srcs: ["a.java"],
1131 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001132 %s
1133 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001134
Colin Cross98be1bb2019-12-13 20:41:13 -08001135 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001136 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1137 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1138 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
Jeongik Cha219141c2020-08-06 23:00:37 +09001139 checkSdkVersion(t, config, test.expectedMinSdkVersion)
Colin Crossd09b0b62018-04-18 11:06:47 -07001140
Colin Crossd09b0b62018-04-18 11:06:47 -07001141 })
1142 }
1143 }
1144}
Colin Crossa4f08812018-10-02 22:03:40 -07001145
Jeongik Cha219141c2020-08-06 23:00:37 +09001146func TestVendorAppSdkVersion(t *testing.T) {
1147 testCases := []struct {
1148 name string
1149 sdkVersion string
1150 platformSdkInt int
1151 platformSdkCodename string
1152 platformSdkFinal bool
1153 deviceCurrentApiLevelForVendorModules string
1154 expectedMinSdkVersion string
1155 }{
1156 {
1157 name: "current final SDK",
1158 sdkVersion: "current",
1159 platformSdkInt: 29,
1160 platformSdkCodename: "REL",
1161 platformSdkFinal: true,
1162 deviceCurrentApiLevelForVendorModules: "29",
1163 expectedMinSdkVersion: "29",
1164 },
1165 {
1166 name: "current final SDK",
1167 sdkVersion: "current",
1168 platformSdkInt: 29,
1169 platformSdkCodename: "REL",
1170 platformSdkFinal: true,
1171 deviceCurrentApiLevelForVendorModules: "28",
1172 expectedMinSdkVersion: "28",
1173 },
1174 {
1175 name: "current final SDK",
1176 sdkVersion: "current",
1177 platformSdkInt: 29,
1178 platformSdkCodename: "Q",
1179 platformSdkFinal: false,
1180 deviceCurrentApiLevelForVendorModules: "current",
1181 expectedMinSdkVersion: "Q",
1182 },
1183 {
1184 name: "current final SDK",
1185 sdkVersion: "current",
1186 platformSdkInt: 29,
1187 platformSdkCodename: "Q",
1188 platformSdkFinal: false,
1189 deviceCurrentApiLevelForVendorModules: "28",
1190 expectedMinSdkVersion: "28",
1191 },
1192 }
1193
1194 for _, moduleType := range []string{"android_app", "android_library"} {
1195 for _, sdkKind := range []string{"", "system_"} {
1196 for _, test := range testCases {
1197 t.Run(moduleType+" "+test.name, func(t *testing.T) {
1198 bp := fmt.Sprintf(`%s {
1199 name: "foo",
1200 srcs: ["a.java"],
1201 sdk_version: "%s%s",
1202 vendor: true,
1203 }`, moduleType, sdkKind, test.sdkVersion)
1204
1205 config := testAppConfig(nil, bp, nil)
1206 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1207 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1208 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1209 config.TestProductVariables.DeviceCurrentApiLevelForVendorModules = &test.deviceCurrentApiLevelForVendorModules
1210 config.TestProductVariables.DeviceSystemSdkVersions = []string{"28", "29"}
1211 checkSdkVersion(t, config, test.expectedMinSdkVersion)
1212 })
1213 }
1214 }
1215 }
1216}
1217
Paul Duffin50c217c2019-06-12 13:25:22 +01001218func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001219 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001220 cc_library {
1221 name: "libjni",
1222 system_shared_libs: [],
Colin Crossc511bc52020-04-07 16:50:32 +00001223 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001224 stl: "none",
1225 }
1226
1227 android_test {
1228 name: "test",
1229 sdk_version: "core_platform",
1230 jni_libs: ["libjni"],
1231 }
1232
1233 android_test {
1234 name: "test_first",
1235 sdk_version: "core_platform",
1236 compile_multilib: "first",
1237 jni_libs: ["libjni"],
1238 }
1239
1240 android_test {
1241 name: "test_both",
1242 sdk_version: "core_platform",
1243 compile_multilib: "both",
1244 jni_libs: ["libjni"],
1245 }
1246
1247 android_test {
1248 name: "test_32",
1249 sdk_version: "core_platform",
1250 compile_multilib: "32",
1251 jni_libs: ["libjni"],
1252 }
1253
1254 android_test {
1255 name: "test_64",
1256 sdk_version: "core_platform",
1257 compile_multilib: "64",
1258 jni_libs: ["libjni"],
1259 }
1260 `)
1261
1262 testCases := []struct {
1263 name string
1264 abis []string
1265 }{
1266 {"test", []string{"arm64-v8a"}},
1267 {"test_first", []string{"arm64-v8a"}},
1268 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1269 {"test_32", []string{"armeabi-v7a"}},
1270 {"test_64", []string{"arm64-v8a"}},
1271 }
1272
1273 for _, test := range testCases {
1274 t.Run(test.name, func(t *testing.T) {
1275 app := ctx.ModuleForTests(test.name, "android_common")
1276 jniLibZip := app.Output("jnilibs.zip")
1277 var abis []string
1278 args := strings.Fields(jniLibZip.Args["jarArgs"])
1279 for i := 0; i < len(args); i++ {
1280 if args[i] == "-P" {
1281 abis = append(abis, filepath.Base(args[i+1]))
1282 i++
1283 }
1284 }
1285 if !reflect.DeepEqual(abis, test.abis) {
1286 t.Errorf("want abis %v, got %v", test.abis, abis)
1287 }
1288 })
1289 }
1290}
1291
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001292func TestAppSdkVersionByPartition(t *testing.T) {
1293 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1294 android_app {
1295 name: "foo",
1296 srcs: ["a.java"],
1297 vendor: true,
1298 platform_apis: true,
1299 }
1300 `)
1301
1302 testJava(t, `
1303 android_app {
1304 name: "bar",
1305 srcs: ["b.java"],
1306 platform_apis: true,
1307 }
1308 `)
1309
1310 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001311 bp := `
1312 android_app {
1313 name: "foo",
1314 srcs: ["a.java"],
1315 product_specific: true,
1316 platform_apis: true,
1317 }
1318 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001319
1320 config := testAppConfig(nil, bp, nil)
1321 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001322 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001323 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 +09001324 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001325 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001326 }
1327 }
1328}
1329
Paul Duffin50c217c2019-06-12 13:25:22 +01001330func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001331 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001332 cc_library {
1333 name: "libjni",
1334 system_shared_libs: [],
1335 stl: "none",
Colin Cross094cde42020-02-15 10:38:00 -08001336 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001337 }
1338
1339 android_app {
1340 name: "app",
1341 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001342 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001343 }
1344
1345 android_app {
1346 name: "app_noembed",
1347 jni_libs: ["libjni"],
1348 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001349 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001350 }
1351
1352 android_app {
1353 name: "app_embed",
1354 jni_libs: ["libjni"],
1355 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001356 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001357 }
1358
1359 android_test {
1360 name: "test",
Colin Crossc511bc52020-04-07 16:50:32 +00001361 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001362 jni_libs: ["libjni"],
1363 }
1364
1365 android_test {
1366 name: "test_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001367 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001368 jni_libs: ["libjni"],
1369 use_embedded_native_libs: false,
1370 }
1371
1372 android_test_helper_app {
1373 name: "test_helper",
Colin Crossc511bc52020-04-07 16:50:32 +00001374 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001375 jni_libs: ["libjni"],
1376 }
1377
1378 android_test_helper_app {
1379 name: "test_helper_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001380 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001381 jni_libs: ["libjni"],
1382 use_embedded_native_libs: false,
1383 }
1384 `)
1385
1386 testCases := []struct {
1387 name string
1388 packaged bool
1389 compressed bool
1390 }{
1391 {"app", false, false},
1392 {"app_noembed", false, false},
1393 {"app_embed", true, false},
1394 {"test", true, false},
1395 {"test_noembed", true, true},
1396 {"test_helper", true, false},
1397 {"test_helper_noembed", true, true},
1398 }
1399
1400 for _, test := range testCases {
1401 t.Run(test.name, func(t *testing.T) {
1402 app := ctx.ModuleForTests(test.name, "android_common")
1403 jniLibZip := app.MaybeOutput("jnilibs.zip")
1404 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1405 t.Errorf("expected jni packaged %v, got %v", w, g)
1406 }
1407
1408 if jniLibZip.Rule != nil {
1409 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1410 t.Errorf("expected jni compressed %v, got %v", w, g)
1411 }
Colin Crossc511bc52020-04-07 16:50:32 +00001412
1413 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1414 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1415 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001416 }
1417 })
1418 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001419}
1420
Colin Cross3c007702020-05-08 11:20:24 -07001421func TestJNISDK(t *testing.T) {
1422 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1423 cc_library {
1424 name: "libjni",
1425 system_shared_libs: [],
1426 stl: "none",
1427 sdk_version: "current",
1428 }
1429
1430 android_test {
1431 name: "app_platform",
1432 jni_libs: ["libjni"],
1433 platform_apis: true,
1434 }
1435
1436 android_test {
1437 name: "app_sdk",
1438 jni_libs: ["libjni"],
1439 sdk_version: "current",
1440 }
1441
1442 android_test {
1443 name: "app_force_platform",
1444 jni_libs: ["libjni"],
1445 sdk_version: "current",
1446 jni_uses_platform_apis: true,
1447 }
1448
1449 android_test {
1450 name: "app_force_sdk",
1451 jni_libs: ["libjni"],
1452 platform_apis: true,
1453 jni_uses_sdk_apis: true,
1454 }
Colin Crossc2d24052020-05-13 11:05:02 -07001455
1456 cc_library {
1457 name: "libvendorjni",
1458 system_shared_libs: [],
1459 stl: "none",
1460 vendor: true,
1461 }
1462
1463 android_test {
1464 name: "app_vendor",
1465 jni_libs: ["libvendorjni"],
1466 sdk_version: "current",
1467 vendor: true,
1468 }
Colin Cross3c007702020-05-08 11:20:24 -07001469 `)
1470
1471 testCases := []struct {
Colin Crossc2d24052020-05-13 11:05:02 -07001472 name string
1473 sdkJNI bool
1474 vendorJNI bool
Colin Cross3c007702020-05-08 11:20:24 -07001475 }{
Colin Crossc2d24052020-05-13 11:05:02 -07001476 {name: "app_platform"},
1477 {name: "app_sdk", sdkJNI: true},
1478 {name: "app_force_platform"},
1479 {name: "app_force_sdk", sdkJNI: true},
1480 {name: "app_vendor", vendorJNI: true},
Colin Cross3c007702020-05-08 11:20:24 -07001481 }
1482
Colin Crossc2d24052020-05-13 11:05:02 -07001483 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1484 Output("libjni.so").Output.String()
1485 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1486 Output("libjni.so").Output.String()
1487 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1488 Output("libvendorjni.so").Output.String()
1489
Colin Cross3c007702020-05-08 11:20:24 -07001490 for _, test := range testCases {
1491 t.Run(test.name, func(t *testing.T) {
1492 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross3c007702020-05-08 11:20:24 -07001493
1494 jniLibZip := app.MaybeOutput("jnilibs.zip")
1495 if len(jniLibZip.Implicits) != 1 {
1496 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1497 }
1498 gotJNI := jniLibZip.Implicits[0].String()
1499
1500 if test.sdkJNI {
1501 if gotJNI != sdkJNI {
1502 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1503 }
Colin Crossc2d24052020-05-13 11:05:02 -07001504 } else if test.vendorJNI {
1505 if gotJNI != vendorJNI {
1506 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1507 }
Colin Cross3c007702020-05-08 11:20:24 -07001508 } else {
1509 if gotJNI != platformJNI {
1510 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1511 }
1512 }
1513 })
1514 }
1515
1516 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1517 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1518 android_test {
1519 name: "app_platform",
1520 platform_apis: true,
1521 jni_uses_platform_apis: true,
1522 }
1523 `)
1524 })
1525
1526 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1527 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1528 android_test {
1529 name: "app_sdk",
1530 sdk_version: "current",
1531 jni_uses_sdk_apis: true,
1532 }
1533 `)
1534 })
1535
1536}
1537
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001538func TestCertificates(t *testing.T) {
1539 testCases := []struct {
1540 name string
1541 bp string
1542 certificateOverride string
Liz Kammere2b27f42020-05-07 13:24:05 -07001543 expectedLineage string
1544 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001545 }{
1546 {
1547 name: "default",
1548 bp: `
1549 android_app {
1550 name: "foo",
1551 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001552 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001553 }
1554 `,
1555 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001556 expectedLineage: "",
1557 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001558 },
1559 {
1560 name: "module certificate property",
1561 bp: `
1562 android_app {
1563 name: "foo",
1564 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001565 certificate: ":new_certificate",
1566 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001567 }
1568
1569 android_app_certificate {
1570 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001571 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001572 }
1573 `,
1574 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001575 expectedLineage: "",
1576 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001577 },
1578 {
1579 name: "path certificate property",
1580 bp: `
1581 android_app {
1582 name: "foo",
1583 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001584 certificate: "expiredkey",
1585 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001586 }
1587 `,
1588 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001589 expectedLineage: "",
1590 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001591 },
1592 {
1593 name: "certificate overrides",
1594 bp: `
1595 android_app {
1596 name: "foo",
1597 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001598 certificate: "expiredkey",
1599 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001600 }
1601
1602 android_app_certificate {
1603 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001604 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001605 }
1606 `,
1607 certificateOverride: "foo:new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001608 expectedLineage: "",
1609 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1610 },
1611 {
1612 name: "certificate lineage",
1613 bp: `
1614 android_app {
1615 name: "foo",
1616 srcs: ["a.java"],
1617 certificate: ":new_certificate",
1618 lineage: "lineage.bin",
1619 sdk_version: "current",
1620 }
1621
1622 android_app_certificate {
1623 name: "new_certificate",
1624 certificate: "cert/new_cert",
1625 }
1626 `,
1627 certificateOverride: "",
1628 expectedLineage: "--lineage lineage.bin",
1629 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001630 },
1631 }
1632
1633 for _, test := range testCases {
1634 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001635 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001636 if test.certificateOverride != "" {
1637 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1638 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001639 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001640
1641 run(t, ctx, config)
1642 foo := ctx.ModuleForTests("foo", "android_common")
1643
1644 signapk := foo.Output("foo.apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001645 signCertificateFlags := signapk.Args["certificates"]
1646 if test.expectedCertificate != signCertificateFlags {
1647 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1648 }
1649
1650 signFlags := signapk.Args["flags"]
1651 if test.expectedLineage != signFlags {
1652 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001653 }
1654 })
1655 }
1656}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001657
1658func TestPackageNameOverride(t *testing.T) {
1659 testCases := []struct {
1660 name string
1661 bp string
1662 packageNameOverride string
1663 expected []string
1664 }{
1665 {
1666 name: "default",
1667 bp: `
1668 android_app {
1669 name: "foo",
1670 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001671 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001672 }
1673 `,
1674 packageNameOverride: "",
1675 expected: []string{
1676 buildDir + "/.intermediates/foo/android_common/foo.apk",
1677 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1678 },
1679 },
1680 {
1681 name: "overridden",
1682 bp: `
1683 android_app {
1684 name: "foo",
1685 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001686 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001687 }
1688 `,
1689 packageNameOverride: "foo:bar",
1690 expected: []string{
1691 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001692 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001693 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1694 },
1695 },
1696 }
1697
1698 for _, test := range testCases {
1699 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001700 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001701 if test.packageNameOverride != "" {
1702 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1703 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001704 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001705
1706 run(t, ctx, config)
1707 foo := ctx.ModuleForTests("foo", "android_common")
1708
1709 outputs := foo.AllOutputs()
1710 outputMap := make(map[string]bool)
1711 for _, o := range outputs {
1712 outputMap[o] = true
1713 }
1714 for _, e := range test.expected {
1715 if _, exist := outputMap[e]; !exist {
1716 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1717 }
1718 }
1719 })
1720 }
1721}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001722
1723func TestInstrumentationTargetOverridden(t *testing.T) {
1724 bp := `
1725 android_app {
1726 name: "foo",
1727 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001728 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001729 }
1730
1731 android_test {
1732 name: "bar",
1733 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001734 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001735 }
1736 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001737 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001738 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001739 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001740
1741 run(t, ctx, config)
1742
1743 bar := ctx.ModuleForTests("bar", "android_common")
1744 res := bar.Output("package-res.apk")
1745 aapt2Flags := res.Args["flags"]
1746 e := "--rename-instrumentation-target-package org.dandroid.bp"
1747 if !strings.Contains(aapt2Flags, e) {
1748 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1749 }
1750}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001751
1752func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001753 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001754 android_app {
1755 name: "foo",
1756 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001757 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001758 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001759 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001760 }
1761
1762 override_android_app {
1763 name: "bar",
1764 base: "foo",
1765 certificate: ":new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001766 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001767 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001768 }
1769
1770 android_app_certificate {
1771 name: "new_certificate",
1772 certificate: "cert/new_cert",
1773 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001774
1775 override_android_app {
1776 name: "baz",
1777 base: "foo",
1778 package_name: "org.dandroid.bp",
1779 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001780 `)
1781
1782 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001783 moduleName string
1784 variantName string
1785 apkName string
1786 apkPath string
Liz Kammere2b27f42020-05-07 13:24:05 -07001787 certFlag string
1788 lineageFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001789 overrides []string
Liz Kammer1d5983b2020-05-19 19:15:37 +00001790 aaptFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001791 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001792 }{
1793 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001794 moduleName: "foo",
1795 variantName: "android_common",
1796 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001797 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1798 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001799 overrides: []string{"qux"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001800 aaptFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001801 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001802 },
1803 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001804 moduleName: "bar",
1805 variantName: "android_common_bar",
1806 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001807 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1808 lineageFlag: "--lineage lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001809 overrides: []string{"qux", "foo"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001810 aaptFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001811 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001812 },
1813 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001814 moduleName: "baz",
1815 variantName: "android_common_baz",
1816 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Liz Kammere2b27f42020-05-07 13:24:05 -07001817 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1818 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001819 overrides: []string{"qux", "foo"},
Liz Kammer1d5983b2020-05-19 19:15:37 +00001820 aaptFlag: "--rename-manifest-package org.dandroid.bp",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001821 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001822 },
1823 }
1824 for _, expected := range expectedVariants {
1825 variant := ctx.ModuleForTests("foo", expected.variantName)
1826
1827 // Check the final apk name
1828 outputs := variant.AllOutputs()
1829 expectedApkPath := buildDir + expected.apkPath
1830 found := false
1831 for _, o := range outputs {
1832 if o == expectedApkPath {
1833 found = true
1834 break
1835 }
1836 }
1837 if !found {
1838 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1839 }
1840
1841 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001842 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001843 certFlag := signapk.Args["certificates"]
1844 if expected.certFlag != certFlag {
1845 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1846 }
1847
1848 // Check the lineage flags
1849 lineageFlag := signapk.Args["flags"]
1850 if expected.lineageFlag != lineageFlag {
1851 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001852 }
1853
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001854 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001855 mod := variant.Module().(*AndroidApp)
1856 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1857 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1858 expected.overrides, mod.appProperties.Overrides)
1859 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001860
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001861 // Test Overridable property: Logging_parent
1862 logging_parent := mod.aapt.LoggingParent
1863 if expected.logging_parent != logging_parent {
1864 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1865 expected.logging_parent, logging_parent)
1866 }
1867
Liz Kammer1d5983b2020-05-19 19:15:37 +00001868 // Check the package renaming flag, if exists.
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001869 res := variant.Output("package-res.apk")
1870 aapt2Flags := res.Args["flags"]
Liz Kammer1d5983b2020-05-19 19:15:37 +00001871 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1872 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1873 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001874 }
1875}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001876
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001877func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001878 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001879 android_app {
1880 name: "foo",
1881 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001882 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001883 }
1884
1885 override_android_app {
1886 name: "bar",
1887 base: "foo",
1888 package_name: "org.dandroid.bp",
1889 }
1890
1891 android_test {
1892 name: "baz",
1893 srcs: ["b.java"],
1894 instrumentation_for: "foo",
1895 }
1896
1897 android_test {
1898 name: "qux",
1899 srcs: ["b.java"],
1900 instrumentation_for: "bar",
1901 }
1902 `)
1903
1904 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1905 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1906 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1907 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1908 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1909 }
1910
1911 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1912 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1913 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1914 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1915 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1916 }
1917}
1918
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001919func TestOverrideAndroidTest(t *testing.T) {
1920 ctx, _ := testJava(t, `
1921 android_app {
1922 name: "foo",
1923 srcs: ["a.java"],
1924 package_name: "com.android.foo",
1925 sdk_version: "current",
1926 }
1927
1928 override_android_app {
1929 name: "bar",
1930 base: "foo",
1931 package_name: "com.android.bar",
1932 }
1933
1934 android_test {
1935 name: "foo_test",
1936 srcs: ["b.java"],
1937 instrumentation_for: "foo",
1938 }
1939
1940 override_android_test {
1941 name: "bar_test",
1942 base: "foo_test",
1943 package_name: "com.android.bar.test",
1944 instrumentation_for: "bar",
1945 instrumentation_target_package: "com.android.bar",
1946 }
1947 `)
1948
1949 expectedVariants := []struct {
1950 moduleName string
1951 variantName string
1952 apkPath string
1953 overrides []string
1954 targetVariant string
1955 packageFlag string
1956 targetPackageFlag string
1957 }{
1958 {
1959 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001960 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001961 overrides: nil,
1962 targetVariant: "android_common",
1963 packageFlag: "",
1964 targetPackageFlag: "",
1965 },
1966 {
1967 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001968 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001969 overrides: []string{"foo_test"},
1970 targetVariant: "android_common_bar",
1971 packageFlag: "com.android.bar.test",
1972 targetPackageFlag: "com.android.bar",
1973 },
1974 }
1975 for _, expected := range expectedVariants {
1976 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1977
1978 // Check the final apk name
1979 outputs := variant.AllOutputs()
1980 expectedApkPath := buildDir + expected.apkPath
1981 found := false
1982 for _, o := range outputs {
1983 if o == expectedApkPath {
1984 found = true
1985 break
1986 }
1987 }
1988 if !found {
1989 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1990 }
1991
1992 // Check if the overrides field values are correctly aggregated.
1993 mod := variant.Module().(*AndroidTest)
1994 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1995 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1996 expected.overrides, mod.appProperties.Overrides)
1997 }
1998
1999 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
2000 javac := variant.Rule("javac")
2001 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
2002 if !strings.Contains(javac.Args["classpath"], turbine) {
2003 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
2004 }
2005
2006 // Check aapt2 flags.
2007 res := variant.Output("package-res.apk")
2008 aapt2Flags := res.Args["flags"]
2009 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
2010 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
2011 }
2012}
2013
Jaewoong Jung39982342020-01-14 10:27:18 -08002014func TestAndroidTest_FixTestConfig(t *testing.T) {
2015 ctx, _ := testJava(t, `
2016 android_app {
2017 name: "foo",
2018 srcs: ["a.java"],
2019 package_name: "com.android.foo",
2020 sdk_version: "current",
2021 }
2022
2023 android_test {
2024 name: "foo_test",
2025 srcs: ["b.java"],
2026 instrumentation_for: "foo",
2027 }
2028
2029 android_test {
2030 name: "bar_test",
2031 srcs: ["b.java"],
2032 package_name: "com.android.bar.test",
2033 instrumentation_for: "foo",
2034 }
2035
2036 override_android_test {
2037 name: "baz_test",
2038 base: "foo_test",
2039 package_name: "com.android.baz.test",
2040 }
2041 `)
2042
2043 testCases := []struct {
2044 moduleName string
2045 variantName string
2046 expectedFlags []string
2047 }{
2048 {
2049 moduleName: "foo_test",
2050 variantName: "android_common",
2051 },
2052 {
2053 moduleName: "bar_test",
2054 variantName: "android_common",
2055 expectedFlags: []string{
2056 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
2057 "--package-name com.android.bar.test",
2058 },
2059 },
2060 {
2061 moduleName: "foo_test",
2062 variantName: "android_common_baz_test",
2063 expectedFlags: []string{
2064 "--manifest " + buildDir +
2065 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
2066 "--package-name com.android.baz.test",
2067 "--test-file-name baz_test.apk",
2068 },
2069 },
2070 }
2071
2072 for _, test := range testCases {
2073 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
2074 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
2075
2076 if len(test.expectedFlags) > 0 {
2077 if params.Rule == nil {
2078 t.Errorf("test_config_fixer was expected to run, but didn't")
2079 } else {
2080 for _, flag := range test.expectedFlags {
2081 if !strings.Contains(params.RuleParams.Command, flag) {
2082 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2083 }
2084 }
2085 }
2086 } else {
2087 if params.Rule != nil {
2088 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2089 }
2090 }
2091
2092 }
2093}
2094
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002095func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002096 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002097 android_app_import {
2098 name: "foo",
2099 apk: "prebuilts/apk/app.apk",
2100 certificate: "platform",
2101 dex_preopt: {
2102 enabled: true,
2103 },
2104 }
2105 `)
2106
2107 variant := ctx.ModuleForTests("foo", "android_common")
2108
2109 // Check dexpreopt outputs.
2110 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2111 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2112 t.Errorf("can't find dexpreopt outputs")
2113 }
2114
2115 // Check cert signing flag.
2116 signedApk := variant.Output("signed/foo.apk")
2117 signingFlag := signedApk.Args["certificates"]
2118 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2119 if expected != signingFlag {
2120 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2121 }
2122}
2123
2124func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002125 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002126 android_app_import {
2127 name: "foo",
2128 apk: "prebuilts/apk/app.apk",
2129 certificate: "platform",
2130 dex_preopt: {
2131 enabled: false,
2132 },
2133 }
2134 `)
2135
2136 variant := ctx.ModuleForTests("foo", "android_common")
2137
2138 // Check dexpreopt outputs. They shouldn't exist.
2139 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2140 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2141 t.Errorf("dexpreopt shouldn't have run.")
2142 }
2143}
2144
2145func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002146 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002147 android_app_import {
2148 name: "foo",
2149 apk: "prebuilts/apk/app.apk",
2150 presigned: true,
2151 dex_preopt: {
2152 enabled: true,
2153 },
2154 }
2155 `)
2156
2157 variant := ctx.ModuleForTests("foo", "android_common")
2158
2159 // Check dexpreopt outputs.
2160 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2161 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2162 t.Errorf("can't find dexpreopt outputs")
2163 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002164 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002165 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2166 t.Errorf("signing rule shouldn't be included.")
2167 }
2168 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2169 t.Errorf("can't find aligning rule")
2170 }
2171}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002172
Liz Kammer24978992020-05-13 15:49:21 -07002173func TestAndroidAppImport_SigningLineage(t *testing.T) {
2174 ctx, _ := testJava(t, `
2175 android_app_import {
2176 name: "foo",
2177 apk: "prebuilts/apk/app.apk",
2178 certificate: "platform",
2179 lineage: "lineage.bin",
2180 }
2181 `)
2182
2183 variant := ctx.ModuleForTests("foo", "android_common")
2184
2185 // Check cert signing lineage flag.
2186 signedApk := variant.Output("signed/foo.apk")
2187 signingFlag := signedApk.Args["flags"]
2188 expected := "--lineage lineage.bin"
2189 if expected != signingFlag {
2190 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2191 }
2192}
2193
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002194func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2195 ctx, _ := testJava(t, `
2196 android_app_import {
2197 name: "foo",
2198 apk: "prebuilts/apk/app.apk",
2199 default_dev_cert: true,
2200 dex_preopt: {
2201 enabled: true,
2202 },
2203 }
2204 `)
2205
2206 variant := ctx.ModuleForTests("foo", "android_common")
2207
2208 // Check dexpreopt outputs.
2209 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2210 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2211 t.Errorf("can't find dexpreopt outputs")
2212 }
2213
2214 // Check cert signing flag.
2215 signedApk := variant.Output("signed/foo.apk")
2216 signingFlag := signedApk.Args["certificates"]
2217 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2218 if expected != signingFlag {
2219 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2220 }
2221}
2222
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002223func TestAndroidAppImport_DpiVariants(t *testing.T) {
2224 bp := `
2225 android_app_import {
2226 name: "foo",
2227 apk: "prebuilts/apk/app.apk",
2228 dpi_variants: {
2229 xhdpi: {
2230 apk: "prebuilts/apk/app_xhdpi.apk",
2231 },
2232 xxhdpi: {
2233 apk: "prebuilts/apk/app_xxhdpi.apk",
2234 },
2235 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002236 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002237 dex_preopt: {
2238 enabled: true,
2239 },
2240 }
2241 `
2242 testCases := []struct {
2243 name string
2244 aaptPreferredConfig *string
2245 aaptPrebuiltDPI []string
2246 expected string
2247 }{
2248 {
2249 name: "no preferred",
2250 aaptPreferredConfig: nil,
2251 aaptPrebuiltDPI: []string{},
2252 expected: "prebuilts/apk/app.apk",
2253 },
2254 {
2255 name: "AAPTPreferredConfig matches",
2256 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002257 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002258 expected: "prebuilts/apk/app_xhdpi.apk",
2259 },
2260 {
2261 name: "AAPTPrebuiltDPI matches",
2262 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2263 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2264 expected: "prebuilts/apk/app_xxhdpi.apk",
2265 },
2266 {
2267 name: "non-first AAPTPrebuiltDPI matches",
2268 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2269 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2270 expected: "prebuilts/apk/app_xhdpi.apk",
2271 },
2272 {
2273 name: "no matches",
2274 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2275 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2276 expected: "prebuilts/apk/app.apk",
2277 },
2278 }
2279
2280 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2281 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002282 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002283 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2284 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002285 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002286
2287 run(t, ctx, config)
2288
2289 variant := ctx.ModuleForTests("foo", "android_common")
2290 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2291 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2292 if len(matches) != 2 {
2293 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2294 }
2295 if test.expected != matches[1] {
2296 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2297 }
2298 }
2299}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002300
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002301func TestAndroidAppImport_Filename(t *testing.T) {
2302 ctx, config := testJava(t, `
2303 android_app_import {
2304 name: "foo",
2305 apk: "prebuilts/apk/app.apk",
2306 presigned: true,
2307 }
2308
2309 android_app_import {
2310 name: "bar",
2311 apk: "prebuilts/apk/app.apk",
2312 presigned: true,
2313 filename: "bar_sample.apk"
2314 }
2315 `)
2316
2317 testCases := []struct {
2318 name string
2319 expected string
2320 }{
2321 {
2322 name: "foo",
2323 expected: "foo.apk",
2324 },
2325 {
2326 name: "bar",
2327 expected: "bar_sample.apk",
2328 },
2329 }
2330
2331 for _, test := range testCases {
2332 variant := ctx.ModuleForTests(test.name, "android_common")
2333 if variant.MaybeOutput(test.expected).Rule == nil {
2334 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2335 }
2336
2337 a := variant.Module().(*AndroidAppImport)
2338 expectedValues := []string{test.expected}
2339 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002340 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002341 if !reflect.DeepEqual(actualValues, expectedValues) {
2342 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2343 actualValues, expectedValues)
2344 }
2345 }
2346}
2347
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002348func TestAndroidAppImport_ArchVariants(t *testing.T) {
2349 // The test config's target arch is ARM64.
2350 testCases := []struct {
2351 name string
2352 bp string
2353 expected string
2354 }{
2355 {
2356 name: "matching arch",
2357 bp: `
2358 android_app_import {
2359 name: "foo",
2360 apk: "prebuilts/apk/app.apk",
2361 arch: {
2362 arm64: {
2363 apk: "prebuilts/apk/app_arm64.apk",
2364 },
2365 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002366 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002367 dex_preopt: {
2368 enabled: true,
2369 },
2370 }
2371 `,
2372 expected: "prebuilts/apk/app_arm64.apk",
2373 },
2374 {
2375 name: "no matching arch",
2376 bp: `
2377 android_app_import {
2378 name: "foo",
2379 apk: "prebuilts/apk/app.apk",
2380 arch: {
2381 arm: {
2382 apk: "prebuilts/apk/app_arm.apk",
2383 },
2384 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002385 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002386 dex_preopt: {
2387 enabled: true,
2388 },
2389 }
2390 `,
2391 expected: "prebuilts/apk/app.apk",
2392 },
2393 }
2394
2395 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2396 for _, test := range testCases {
2397 ctx, _ := testJava(t, test.bp)
2398
2399 variant := ctx.ModuleForTests("foo", "android_common")
2400 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2401 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2402 if len(matches) != 2 {
2403 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2404 }
2405 if test.expected != matches[1] {
2406 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2407 }
2408 }
2409}
2410
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002411func TestAndroidTestImport(t *testing.T) {
2412 ctx, config := testJava(t, `
2413 android_test_import {
2414 name: "foo",
2415 apk: "prebuilts/apk/app.apk",
2416 presigned: true,
2417 data: [
2418 "testdata/data",
2419 ],
2420 }
2421 `)
2422
2423 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2424
2425 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002426 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002427 expected := []string{"tests"}
2428 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2429 if !reflect.DeepEqual(expected, actual) {
2430 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2431 }
2432 expected = []string{"testdata/data:testdata/data"}
2433 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2434 if !reflect.DeepEqual(expected, actual) {
2435 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2436 }
2437}
2438
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002439func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2440 ctx, _ := testJava(t, `
2441 android_test_import {
2442 name: "foo",
2443 apk: "prebuilts/apk/app.apk",
2444 certificate: "cert/new_cert",
2445 data: [
2446 "testdata/data",
2447 ],
2448 }
2449
2450 android_test_import {
2451 name: "foo_presigned",
2452 apk: "prebuilts/apk/app.apk",
2453 presigned: true,
2454 data: [
2455 "testdata/data",
2456 ],
2457 }
2458 `)
2459
2460 variant := ctx.ModuleForTests("foo", "android_common")
2461 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2462 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2463 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2464 }
2465
2466 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2467 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2468 if jniRule != android.Cp.String() {
2469 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2470 }
Liz Kammer3b70b3f2020-05-20 14:36:30 -07002471 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2472 t.Errorf("Presigned test apk should be aligned")
2473 }
2474}
2475
2476func TestAndroidTestImport_Preprocessed(t *testing.T) {
2477 ctx, _ := testJava(t, `
2478 android_test_import {
2479 name: "foo",
2480 apk: "prebuilts/apk/app.apk",
2481 presigned: true,
2482 preprocessed: true,
2483 }
2484
2485 android_test_import {
2486 name: "foo_cert",
2487 apk: "prebuilts/apk/app.apk",
2488 certificate: "cert/new_cert",
2489 preprocessed: true,
2490 }
2491 `)
2492
2493 testModules := []string{"foo", "foo_cert"}
2494 for _, m := range testModules {
2495 apkName := m + ".apk"
2496 variant := ctx.ModuleForTests(m, "android_common")
2497 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2498 if jniRule != android.Cp.String() {
2499 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2500 }
2501
2502 // Make sure signing and aligning were skipped.
2503 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2504 t.Errorf("signing rule shouldn't be included for preprocessed.")
2505 }
2506 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2507 t.Errorf("aligning rule shouldn't be for preprocessed")
2508 }
2509 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002510}
2511
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002512func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002513 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002514 cc_library {
2515 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002516 sdk_version: "current",
2517 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002518 }
2519
2520 android_test {
2521 name: "stl",
2522 jni_libs: ["libjni"],
2523 compile_multilib: "both",
2524 sdk_version: "current",
2525 stl: "c++_shared",
2526 }
2527
2528 android_test {
2529 name: "system",
2530 jni_libs: ["libjni"],
2531 compile_multilib: "both",
2532 sdk_version: "current",
2533 }
2534 `)
2535
2536 testCases := []struct {
2537 name string
2538 jnis []string
2539 }{
2540 {"stl",
2541 []string{
2542 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002543 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002544 },
2545 },
2546 {"system",
2547 []string{
2548 "libjni.so",
2549 },
2550 },
2551 }
2552
2553 for _, test := range testCases {
2554 t.Run(test.name, func(t *testing.T) {
2555 app := ctx.ModuleForTests(test.name, "android_common")
2556 jniLibZip := app.Output("jnilibs.zip")
2557 var jnis []string
2558 args := strings.Fields(jniLibZip.Args["jarArgs"])
2559 for i := 0; i < len(args); i++ {
2560 if args[i] == "-f" {
2561 jnis = append(jnis, args[i+1])
2562 i += 1
2563 }
2564 }
2565 jnisJoined := strings.Join(jnis, " ")
2566 for _, jni := range test.jnis {
2567 if !strings.Contains(jnisJoined, jni) {
2568 t.Errorf("missing jni %q in %q", jni, jnis)
2569 }
2570 }
2571 })
2572 }
2573}
Colin Cross50ddcc42019-05-16 12:28:22 -07002574
2575func TestUsesLibraries(t *testing.T) {
2576 bp := `
2577 java_sdk_library {
2578 name: "foo",
2579 srcs: ["a.java"],
2580 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002581 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002582 }
2583
2584 java_sdk_library {
Paul Duffin859fe962020-05-15 10:20:31 +01002585 name: "qux",
2586 srcs: ["a.java"],
2587 api_packages: ["qux"],
2588 sdk_version: "current",
2589 }
2590
2591 java_sdk_library {
2592 name: "quuz",
2593 srcs: ["a.java"],
2594 api_packages: ["quuz"],
2595 sdk_version: "current",
2596 }
2597
2598 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002599 name: "bar",
2600 srcs: ["a.java"],
2601 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002602 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002603 }
2604
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002605 java_sdk_library {
2606 name: "runtime-library",
2607 srcs: ["a.java"],
2608 sdk_version: "current",
2609 }
2610
2611 java_library {
2612 name: "static-runtime-helper",
2613 srcs: ["a.java"],
2614 libs: ["runtime-library"],
2615 sdk_version: "current",
2616 }
2617
Colin Cross50ddcc42019-05-16 12:28:22 -07002618 android_app {
2619 name: "app",
2620 srcs: ["a.java"],
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002621 libs: ["qux", "quuz"],
2622 static_libs: ["static-runtime-helper"],
2623 uses_libs: ["foo"],
2624 sdk_version: "current",
2625 optional_uses_libs: [
2626 "bar",
2627 "baz",
2628 ],
2629 }
2630
2631 android_app {
2632 name: "app_with_stub_deps",
2633 srcs: ["a.java"],
Paul Duffin859fe962020-05-15 10:20:31 +01002634 libs: ["qux", "quuz.stubs"],
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002635 static_libs: ["static-runtime-helper"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002636 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002637 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002638 optional_uses_libs: [
2639 "bar",
2640 "baz",
2641 ],
2642 }
2643
2644 android_app_import {
2645 name: "prebuilt",
2646 apk: "prebuilts/apk/app.apk",
2647 certificate: "platform",
2648 uses_libs: ["foo"],
2649 optional_uses_libs: [
2650 "bar",
2651 "baz",
2652 ],
2653 }
2654 `
2655
Colin Cross98be1bb2019-12-13 20:41:13 -08002656 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002657 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2658
Colin Cross98be1bb2019-12-13 20:41:13 -08002659 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002660
2661 run(t, ctx, config)
2662
2663 app := ctx.ModuleForTests("app", "android_common")
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002664 appWithStubDeps := ctx.ModuleForTests("app_with_stub_deps", "android_common")
Colin Cross50ddcc42019-05-16 12:28:22 -07002665 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2666
Paul Duffin859fe962020-05-15 10:20:31 +01002667 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2668 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002669 for _, w := range []string{"qux", "quuz", "runtime-library"} {
2670 if !strings.Contains(manifestFixerArgs, "--uses-library "+w) {
2671 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2672 }
Paul Duffin859fe962020-05-15 10:20:31 +01002673 }
2674
Colin Cross50ddcc42019-05-16 12:28:22 -07002675 // Test that all libraries are verified
2676 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2677 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2678 t.Errorf("wanted %q in %q", w, cmd)
2679 }
2680
2681 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2682 t.Errorf("wanted %q in %q", w, cmd)
2683 }
2684
2685 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2686
2687 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2688 t.Errorf("wanted %q in %q", w, cmd)
2689 }
2690
2691 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2692 t.Errorf("wanted %q in %q", w, cmd)
2693 }
2694
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002695 // Test that all present libraries are preopted, including implicit SDK dependencies
Colin Cross50ddcc42019-05-16 12:28:22 -07002696 cmd = app.Rule("dexpreopt").RuleParams.Command
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002697 w := `--target-classpath-for-sdk any` +
2698 ` /system/framework/foo.jar` +
2699 `:/system/framework/quuz.jar` +
2700 `:/system/framework/qux.jar` +
2701 `:/system/framework/runtime-library.jar` +
2702 `:/system/framework/bar.jar`
2703 if !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002704 t.Errorf("wanted %q in %q", w, cmd)
2705 }
2706
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002707 // TODO(skvadrik) fix dexpreopt for stub libraries for which the implementation is present
2708 if appWithStubDeps.MaybeRule("dexpreopt").RuleParams.Command != "" {
2709 t.Errorf("dexpreopt should be disabled for apps with dependencies on stub libraries")
2710 }
Colin Cross50ddcc42019-05-16 12:28:22 -07002711
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002712 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
Ulya Trafimovich5f364b62020-06-30 12:39:01 +01002713 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 -07002714 t.Errorf("wanted %q in %q", w, cmd)
2715 }
2716}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002717
2718func TestCodelessApp(t *testing.T) {
2719 testCases := []struct {
2720 name string
2721 bp string
2722 noCode bool
2723 }{
2724 {
2725 name: "normal",
2726 bp: `
2727 android_app {
2728 name: "foo",
2729 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002730 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002731 }
2732 `,
2733 noCode: false,
2734 },
2735 {
2736 name: "app without sources",
2737 bp: `
2738 android_app {
2739 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002740 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002741 }
2742 `,
2743 noCode: true,
2744 },
2745 {
2746 name: "app with libraries",
2747 bp: `
2748 android_app {
2749 name: "foo",
2750 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002751 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002752 }
2753
2754 java_library {
2755 name: "lib",
2756 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002757 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002758 }
2759 `,
2760 noCode: false,
2761 },
2762 {
2763 name: "app with sourceless libraries",
2764 bp: `
2765 android_app {
2766 name: "foo",
2767 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002768 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002769 }
2770
2771 java_library {
2772 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002773 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002774 }
2775 `,
2776 // TODO(jungjw): this should probably be true
2777 noCode: false,
2778 },
2779 }
2780
2781 for _, test := range testCases {
2782 t.Run(test.name, func(t *testing.T) {
2783 ctx := testApp(t, test.bp)
2784
2785 foo := ctx.ModuleForTests("foo", "android_common")
2786 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2787 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2788 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2789 }
2790 })
2791 }
2792}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002793
2794func TestEmbedNotice(t *testing.T) {
Colin Cross238c1f32020-06-07 16:58:18 -07002795 ctx, _ := testJavaWithFS(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002796 android_app {
2797 name: "foo",
2798 srcs: ["a.java"],
2799 static_libs: ["javalib"],
2800 jni_libs: ["libjni"],
2801 notice: "APP_NOTICE",
2802 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002803 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002804 }
2805
2806 // No embed_notice flag
2807 android_app {
2808 name: "bar",
2809 srcs: ["a.java"],
2810 jni_libs: ["libjni"],
2811 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002812 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002813 }
2814
2815 // No NOTICE files
2816 android_app {
2817 name: "baz",
2818 srcs: ["a.java"],
2819 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002820 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002821 }
2822
2823 cc_library {
2824 name: "libjni",
2825 system_shared_libs: [],
2826 stl: "none",
2827 notice: "LIB_NOTICE",
Colin Cross094cde42020-02-15 10:38:00 -08002828 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002829 }
2830
2831 java_library {
2832 name: "javalib",
2833 srcs: [
2834 ":gen",
2835 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002836 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002837 }
2838
2839 genrule {
2840 name: "gen",
2841 tools: ["gentool"],
2842 out: ["gen.java"],
2843 notice: "GENRULE_NOTICE",
2844 }
2845
2846 java_binary_host {
2847 name: "gentool",
2848 srcs: ["b.java"],
2849 notice: "TOOL_NOTICE",
2850 }
Colin Cross238c1f32020-06-07 16:58:18 -07002851 `, map[string][]byte{
2852 "APP_NOTICE": nil,
2853 "GENRULE_NOTICE": nil,
2854 "LIB_NOTICE": nil,
2855 "TOOL_NOTICE": nil,
2856 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002857
2858 // foo has NOTICE files to process, and embed_notices is true.
2859 foo := ctx.ModuleForTests("foo", "android_common")
2860 // verify merge notices rule.
2861 mergeNotices := foo.Rule("mergeNoticesRule")
2862 noticeInputs := mergeNotices.Inputs.Strings()
2863 // TOOL_NOTICE should be excluded as it's a host module.
2864 if len(mergeNotices.Inputs) != 3 {
2865 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2866 }
2867 if !inList("APP_NOTICE", noticeInputs) {
2868 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2869 }
2870 if !inList("LIB_NOTICE", noticeInputs) {
2871 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2872 }
2873 if !inList("GENRULE_NOTICE", noticeInputs) {
2874 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2875 }
2876 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2877 res := foo.Output("package-res.apk")
2878 aapt2Flags := res.Args["flags"]
2879 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2880 if !strings.Contains(aapt2Flags, e) {
2881 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2882 }
2883
2884 // bar has NOTICE files to process, but embed_notices is not set.
2885 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002886 res = bar.Output("package-res.apk")
2887 aapt2Flags = res.Args["flags"]
2888 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2889 if strings.Contains(aapt2Flags, e) {
2890 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002891 }
2892
2893 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2894 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002895 res = baz.Output("package-res.apk")
2896 aapt2Flags = res.Args["flags"]
2897 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2898 if strings.Contains(aapt2Flags, e) {
2899 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002900 }
2901}
Colin Cross53a87f52019-06-25 13:35:30 -07002902
2903func TestUncompressDex(t *testing.T) {
2904 testCases := []struct {
2905 name string
2906 bp string
2907
2908 uncompressedPlatform bool
2909 uncompressedUnbundled bool
2910 }{
2911 {
2912 name: "normal",
2913 bp: `
2914 android_app {
2915 name: "foo",
2916 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002917 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002918 }
2919 `,
2920 uncompressedPlatform: true,
2921 uncompressedUnbundled: false,
2922 },
2923 {
2924 name: "use_embedded_dex",
2925 bp: `
2926 android_app {
2927 name: "foo",
2928 use_embedded_dex: true,
2929 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002930 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002931 }
2932 `,
2933 uncompressedPlatform: true,
2934 uncompressedUnbundled: true,
2935 },
2936 {
2937 name: "privileged",
2938 bp: `
2939 android_app {
2940 name: "foo",
2941 privileged: true,
2942 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002943 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002944 }
2945 `,
2946 uncompressedPlatform: true,
2947 uncompressedUnbundled: true,
2948 },
David Srbeckye033cba2020-05-20 22:20:28 +01002949 {
2950 name: "normal_uncompress_dex_true",
2951 bp: `
2952 android_app {
2953 name: "foo",
2954 srcs: ["a.java"],
2955 sdk_version: "current",
2956 uncompress_dex: true,
2957 }
2958 `,
2959 uncompressedPlatform: true,
2960 uncompressedUnbundled: true,
2961 },
2962 {
2963 name: "normal_uncompress_dex_false",
2964 bp: `
2965 android_app {
2966 name: "foo",
2967 srcs: ["a.java"],
2968 sdk_version: "current",
2969 uncompress_dex: false,
2970 }
2971 `,
2972 uncompressedPlatform: false,
2973 uncompressedUnbundled: false,
2974 },
Colin Cross53a87f52019-06-25 13:35:30 -07002975 }
2976
2977 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2978 t.Helper()
2979
Colin Cross98be1bb2019-12-13 20:41:13 -08002980 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002981 if unbundled {
2982 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
Jeongik Cha816a23a2020-07-08 01:09:23 +09002983 config.TestProductVariables.Always_use_prebuilt_sdks = proptools.BoolPtr(true)
Colin Cross53a87f52019-06-25 13:35:30 -07002984 }
2985
Colin Cross98be1bb2019-12-13 20:41:13 -08002986 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002987
2988 run(t, ctx, config)
2989
2990 foo := ctx.ModuleForTests("foo", "android_common")
2991 dex := foo.Rule("r8")
2992 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2993 aligned := foo.MaybeRule("zipalign").Rule != nil
2994
2995 if uncompressedInDexJar != want {
2996 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2997 }
2998
2999 if aligned != want {
3000 t.Errorf("want aligned %v, got %v", want, aligned)
3001 }
3002 }
3003
3004 for _, tt := range testCases {
3005 t.Run(tt.name, func(t *testing.T) {
3006 t.Run("platform", func(t *testing.T) {
3007 test(t, tt.bp, tt.uncompressedPlatform, false)
3008 })
3009 t.Run("unbundled", func(t *testing.T) {
3010 test(t, tt.bp, tt.uncompressedUnbundled, true)
3011 })
3012 })
3013 }
3014}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07003015
3016func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
3017 if expectedValue != "" {
3018 expectedFlag := "--" + flagName + " " + expectedValue
3019 if !strings.Contains(aapt2Flags, expectedFlag) {
3020 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
3021 }
3022 } else {
3023 unexpectedFlag := "--" + flagName
3024 if strings.Contains(aapt2Flags, unexpectedFlag) {
3025 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
3026 }
3027 }
3028}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003029
3030func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003031 fs := map[string][]byte{
3032 "baz/res/res/values/strings.xml": nil,
3033 "bar/res/res/values/strings.xml": nil,
3034 }
3035 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003036 runtime_resource_overlay {
3037 name: "foo",
3038 certificate: "platform",
Liz Kammer966b2f02020-05-19 16:15:25 -07003039 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003040 product_specific: true,
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003041 static_libs: ["bar"],
3042 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003043 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003044 }
3045
3046 runtime_resource_overlay {
3047 name: "foo_themed",
3048 certificate: "platform",
3049 product_specific: true,
3050 theme: "faza",
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003051 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003052 }
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003053
3054 android_library {
3055 name: "bar",
3056 resource_dirs: ["bar/res"],
3057 }
3058
3059 android_app {
3060 name: "baz",
3061 sdk_version: "current",
3062 resource_dirs: ["baz/res"],
3063 }
3064 `
3065 config := testAppConfig(nil, bp, fs)
3066 ctx := testContext()
3067 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003068
3069 m := ctx.ModuleForTests("foo", "android_common")
3070
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003071 // Check AAPT2 link flags.
3072 aapt2Flags := m.Output("package-res.apk").Args["flags"]
3073 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3074 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
3075 if len(absentFlags) > 0 {
3076 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3077 }
3078
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003079 // Check overlay.list output for static_libs dependency.
3080 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
3081 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
3082 if !inList(staticLibPackage, overlayList) {
3083 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
3084 }
3085
3086 // Check AAPT2 link flags for resource_libs dependency.
3087 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
3088 if !strings.Contains(aapt2Flags, resourceLibFlag) {
3089 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
3090 }
3091
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003092 // Check cert signing flag.
3093 signedApk := m.Output("signed/foo.apk")
Liz Kammer966b2f02020-05-19 16:15:25 -07003094 lineageFlag := signedApk.Args["flags"]
3095 expectedLineageFlag := "--lineage lineage.bin"
3096 if expectedLineageFlag != lineageFlag {
3097 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
3098 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003099 signingFlag := signedApk.Args["certificates"]
3100 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
3101 if expected != signingFlag {
3102 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
3103 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003104 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3105 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003106 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
3107 if !reflect.DeepEqual(path, expectedPath) {
3108 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
3109 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003110
3111 // Check device location.
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003112 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003113 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003114 if !reflect.DeepEqual(path, expectedPath) {
3115 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3116 }
3117
3118 // A themed module has a different device location
3119 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003120 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3121 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003122 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3123 if !reflect.DeepEqual(path, expectedPath) {
3124 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3125 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003126
3127 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3128 expectedOverrides := []string{"foo"}
3129 if !reflect.DeepEqual(overrides, expectedOverrides) {
3130 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3131 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003132}
Roshan Pius4df2bc72020-04-27 09:42:27 -07003133
3134func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3135 ctx, _ := testJava(t, `
3136 runtime_resource_overlay {
3137 name: "foo_overlay",
3138 certificate: "platform",
3139 product_specific: true,
3140 sdk_version: "current",
3141 }
3142
3143 override_runtime_resource_overlay {
3144 name: "bar_overlay",
3145 base: "foo_overlay",
3146 package_name: "com.android.bar.overlay",
3147 target_package_name: "com.android.bar",
3148 }
3149 `)
3150
3151 expectedVariants := []struct {
3152 moduleName string
3153 variantName string
3154 apkPath string
3155 overrides []string
3156 targetVariant string
3157 packageFlag string
3158 targetPackageFlag string
3159 }{
3160 {
3161 variantName: "android_common",
3162 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3163 overrides: nil,
3164 targetVariant: "android_common",
3165 packageFlag: "",
3166 targetPackageFlag: "",
3167 },
3168 {
3169 variantName: "android_common_bar_overlay",
3170 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3171 overrides: []string{"foo_overlay"},
3172 targetVariant: "android_common_bar",
3173 packageFlag: "com.android.bar.overlay",
3174 targetPackageFlag: "com.android.bar",
3175 },
3176 }
3177 for _, expected := range expectedVariants {
3178 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3179
3180 // Check the final apk name
3181 outputs := variant.AllOutputs()
3182 expectedApkPath := buildDir + expected.apkPath
3183 found := false
3184 for _, o := range outputs {
3185 if o == expectedApkPath {
3186 found = true
3187 break
3188 }
3189 }
3190 if !found {
3191 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3192 }
3193
3194 // Check if the overrides field values are correctly aggregated.
3195 mod := variant.Module().(*RuntimeResourceOverlay)
3196 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3197 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3198 expected.overrides, mod.properties.Overrides)
3199 }
3200
3201 // Check aapt2 flags.
3202 res := variant.Output("package-res.apk")
3203 aapt2Flags := res.Args["flags"]
3204 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
3205 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3206 }
3207}
Jaewoong Jungbf135462020-04-26 15:10:51 -07003208
3209func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3210 ctx, config := testJava(t, `
3211 java_defaults {
3212 name: "rro_defaults",
3213 theme: "default_theme",
3214 product_specific: true,
3215 aaptflags: ["--keep-raw-values"],
3216 }
3217
3218 runtime_resource_overlay {
3219 name: "foo_with_defaults",
3220 defaults: ["rro_defaults"],
3221 }
3222
3223 runtime_resource_overlay {
3224 name: "foo_barebones",
3225 }
3226 `)
3227
3228 //
3229 // RRO module with defaults
3230 //
3231 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3232
3233 // Check AAPT2 link flags.
3234 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3235 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3236 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3237 if len(absentFlags) > 0 {
3238 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3239 }
3240
3241 // Check device location.
3242 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3243 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3244 if !reflect.DeepEqual(path, expectedPath) {
3245 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3246 }
3247
3248 //
3249 // RRO module without defaults
3250 //
3251 m = ctx.ModuleForTests("foo_barebones", "android_common")
3252
3253 // Check AAPT2 link flags.
3254 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3255 unexpectedFlags := "--keep-raw-values"
3256 if inList(unexpectedFlags, aapt2Flags) {
3257 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3258 }
3259
3260 // Check device location.
3261 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3262 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3263 if !reflect.DeepEqual(path, expectedPath) {
3264 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3265 }
3266}