blob: 6efb0dcb327183bdbc5d0e4b132803e33471ccef [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 Crossae8600b2020-10-29 17:09:13 -070062 ctx := testContext(config)
Colin Cross527012a2017-11-30 22:56:16 -080063
64 run(t, ctx, config)
65
66 return ctx
Colin Cross3bc7ffa2017-11-22 16:19:37 -080067}
68
69func TestApp(t *testing.T) {
Colin Crossa97c5d32018-03-28 14:58:31 -070070 for _, moduleType := range []string{"android_app", "android_library"} {
71 t.Run(moduleType, func(t *testing.T) {
72 ctx := testApp(t, moduleType+` {
73 name: "foo",
74 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +090075 sdk_version: "current"
Colin Crossa97c5d32018-03-28 14:58:31 -070076 }
77 `)
Colin Cross3bc7ffa2017-11-22 16:19:37 -080078
Colin Crossa97c5d32018-03-28 14:58:31 -070079 foo := ctx.ModuleForTests("foo", "android_common")
Colin Cross3bc7ffa2017-11-22 16:19:37 -080080
Colin Cross31656952018-05-24 16:11:20 -070081 var expectedLinkImplicits []string
82
83 manifestFixer := foo.Output("manifest_fixer/AndroidManifest.xml")
84 expectedLinkImplicits = append(expectedLinkImplicits, manifestFixer.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080085
Colin Crossa97c5d32018-03-28 14:58:31 -070086 frameworkRes := ctx.ModuleForTests("framework-res", "android_common")
87 expectedLinkImplicits = append(expectedLinkImplicits,
88 frameworkRes.Output("package-res.apk").Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080089
Colin Crossa97c5d32018-03-28 14:58:31 -070090 // Test the mapping from input files to compiled output file names
91 compile := foo.Output(compiledResourceFiles[0])
92 if !reflect.DeepEqual(resourceFiles, compile.Inputs.Strings()) {
93 t.Errorf("expected aapt2 compile inputs expected:\n %#v\n got:\n %#v",
94 resourceFiles, compile.Inputs.Strings())
95 }
Colin Crossb69301e2017-12-01 10:48:26 -080096
Colin Crossa97c5d32018-03-28 14:58:31 -070097 compiledResourceOutputs := compile.Outputs.Strings()
98 sort.Strings(compiledResourceOutputs)
Colin Crossb69301e2017-12-01 10:48:26 -080099
Colin Crossa97c5d32018-03-28 14:58:31 -0700100 expectedLinkImplicits = append(expectedLinkImplicits, compiledResourceOutputs...)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800101
Colin Crossa97c5d32018-03-28 14:58:31 -0700102 list := foo.Output("aapt2/res.list")
103 expectedLinkImplicits = append(expectedLinkImplicits, list.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800104
Colin Crossa97c5d32018-03-28 14:58:31 -0700105 // Check that the link rule uses
106 res := ctx.ModuleForTests("foo", "android_common").Output("package-res.apk")
107 if !reflect.DeepEqual(expectedLinkImplicits, res.Implicits.Strings()) {
108 t.Errorf("expected aapt2 link implicits expected:\n %#v\n got:\n %#v",
109 expectedLinkImplicits, res.Implicits.Strings())
110 }
111 })
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800112 }
113}
Colin Cross890ff552017-11-30 20:13:19 -0800114
Colin Crosse560c4a2019-03-19 16:03:11 -0700115func TestAppSplits(t *testing.T) {
116 ctx := testApp(t, `
117 android_app {
118 name: "foo",
119 srcs: ["a.java"],
120 package_splits: ["v4", "v7,hdpi"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900121 sdk_version: "current"
Colin Crosse560c4a2019-03-19 16:03:11 -0700122 }`)
123
124 foo := ctx.ModuleForTests("foo", "android_common")
125
126 expectedOutputs := []string{
127 filepath.Join(buildDir, ".intermediates/foo/android_common/foo.apk"),
128 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v4.apk"),
129 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v7_hdpi.apk"),
130 }
131 for _, expectedOutput := range expectedOutputs {
132 foo.Output(expectedOutput)
133 }
134
Colin Cross41955e82019-05-29 14:40:35 -0700135 outputFiles, err := foo.Module().(*AndroidApp).OutputFiles("")
136 if err != nil {
137 t.Fatal(err)
138 }
139 if g, w := outputFiles.Strings(), expectedOutputs; !reflect.DeepEqual(g, w) {
140 t.Errorf(`want OutputFiles("") = %q, got %q`, w, g)
Colin Crosse560c4a2019-03-19 16:03:11 -0700141 }
142}
143
Sasha Smundaka7856c02020-04-23 09:49:59 -0700144func TestAndroidAppSet(t *testing.T) {
145 ctx, config := testJava(t, `
146 android_app_set {
147 name: "foo",
148 set: "prebuilts/apks/app.apks",
149 prerelease: true,
Jaewoong Jung11c1e0f2020-06-29 19:18:44 -0700150 }`)
Sasha Smundaka7856c02020-04-23 09:49:59 -0700151 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundak18d98bc2020-05-27 16:36:07 -0700152 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-04-23 09:49:59 -0700153 params := module.Output(packedSplitApks)
154 if params.Rule == nil {
155 t.Errorf("expected output %s is missing", packedSplitApks)
156 }
157 if s := params.Args["allow-prereleased"]; s != "true" {
158 t.Errorf("wrong allow-prereleased value: '%s', expected 'true'", s)
159 }
Jaewoong Jung11c1e0f2020-06-29 19:18:44 -0700160 if s := params.Args["partition"]; s != "system" {
161 t.Errorf("wrong partition value: '%s', expected 'system'", s)
162 }
Sasha Smundaka7856c02020-04-23 09:49:59 -0700163 mkEntries := android.AndroidMkEntriesForTest(t, config, "", module.Module())[0]
Liz Kammercada8072020-07-28 15:47:38 -0700164 actualInstallFile := mkEntries.EntryMap["LOCAL_APK_SET_INSTALL_FILE"]
165 expectedInstallFile := []string{"foo.apk"}
166 if !reflect.DeepEqual(actualInstallFile, expectedInstallFile) {
167 t.Errorf("Unexpected LOCAL_APK_SET_INSTALL_FILE value: '%s', expected: '%s',",
168 actualInstallFile, expectedInstallFile)
Sasha Smundaka7856c02020-04-23 09:49:59 -0700169 }
170}
171
172func TestAndroidAppSet_Variants(t *testing.T) {
173 bp := `
174 android_app_set {
175 name: "foo",
176 set: "prebuilts/apks/app.apks",
177 }`
178 testCases := []struct {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700179 name string
180 targets []android.Target
181 aaptPrebuiltDPI []string
182 sdkVersion int
183 expected map[string]string
Sasha Smundaka7856c02020-04-23 09:49:59 -0700184 }{
185 {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700186 name: "One",
187 targets: []android.Target{
188 {Os: android.Android, Arch: android.Arch{ArchType: android.X86}},
189 },
Sasha Smundaka7856c02020-04-23 09:49:59 -0700190 aaptPrebuiltDPI: []string{"ldpi", "xxhdpi"},
191 sdkVersion: 29,
192 expected: map[string]string{
193 "abis": "X86",
194 "allow-prereleased": "false",
195 "screen-densities": "LDPI,XXHDPI",
196 "sdk-version": "29",
197 "stem": "foo",
198 },
199 },
200 {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700201 name: "Two",
202 targets: []android.Target{
203 {Os: android.Android, Arch: android.Arch{ArchType: android.X86_64}},
204 {Os: android.Android, Arch: android.Arch{ArchType: android.X86}},
205 },
206 aaptPrebuiltDPI: nil,
207 sdkVersion: 30,
Sasha Smundaka7856c02020-04-23 09:49:59 -0700208 expected: map[string]string{
209 "abis": "X86_64,X86",
210 "allow-prereleased": "false",
211 "screen-densities": "all",
212 "sdk-version": "30",
213 "stem": "foo",
214 },
215 },
216 }
217
218 for _, test := range testCases {
219 config := testAppConfig(nil, bp, nil)
220 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
221 config.TestProductVariables.Platform_sdk_version = &test.sdkVersion
Jaewoong Jung829b7132020-06-10 12:23:32 -0700222 config.Targets[android.Android] = test.targets
Colin Crossae8600b2020-10-29 17:09:13 -0700223 ctx := testContext(config)
Sasha Smundaka7856c02020-04-23 09:49:59 -0700224 run(t, ctx, config)
225 module := ctx.ModuleForTests("foo", "android_common")
Sasha Smundak18d98bc2020-05-27 16:36:07 -0700226 const packedSplitApks = "foo.zip"
Sasha Smundaka7856c02020-04-23 09:49:59 -0700227 params := module.Output(packedSplitApks)
228 for k, v := range test.expected {
229 if actual := params.Args[k]; actual != v {
230 t.Errorf("%s: bad build arg value for '%s': '%s', expected '%s'",
231 test.name, k, actual, v)
232 }
233 }
234 }
235}
236
Jeongik Cha538c0d02019-07-11 15:54:27 +0900237func TestPlatformAPIs(t *testing.T) {
238 testJava(t, `
239 android_app {
240 name: "foo",
241 srcs: ["a.java"],
242 platform_apis: true,
243 }
244 `)
245
246 testJava(t, `
247 android_app {
248 name: "foo",
249 srcs: ["a.java"],
250 sdk_version: "current",
251 }
252 `)
253
254 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
255 android_app {
256 name: "bar",
257 srcs: ["b.java"],
258 }
259 `)
260
261 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
262 android_app {
263 name: "bar",
264 srcs: ["b.java"],
265 sdk_version: "system_current",
266 platform_apis: true,
267 }
268 `)
269}
270
Jeongik Chae403e9e2019-12-07 00:16:24 +0900271func TestAndroidAppLinkType(t *testing.T) {
272 testJava(t, `
273 android_app {
274 name: "foo",
275 srcs: ["a.java"],
276 libs: ["bar"],
277 static_libs: ["baz"],
278 platform_apis: true,
279 }
280
281 java_library {
282 name: "bar",
283 sdk_version: "current",
284 srcs: ["b.java"],
285 }
286
287 android_library {
288 name: "baz",
289 sdk_version: "system_current",
290 srcs: ["c.java"],
291 }
292 `)
293
Steven Moreland00298982020-11-17 21:44:36 +0000294 testJavaError(t, "consider adjusting sdk_version: OR platform_apis:", `
Jeongik Chae403e9e2019-12-07 00:16:24 +0900295 android_app {
296 name: "foo",
297 srcs: ["a.java"],
298 libs: ["bar"],
299 sdk_version: "current",
300 static_libs: ["baz"],
301 }
302
303 java_library {
304 name: "bar",
305 sdk_version: "current",
306 srcs: ["b.java"],
307 }
308
309 android_library {
310 name: "baz",
311 sdk_version: "system_current",
312 srcs: ["c.java"],
313 }
314 `)
315
316 testJava(t, `
317 android_app {
318 name: "foo",
319 srcs: ["a.java"],
320 libs: ["bar"],
321 sdk_version: "system_current",
322 static_libs: ["baz"],
323 }
324
325 java_library {
326 name: "bar",
327 sdk_version: "current",
328 srcs: ["b.java"],
329 }
330
331 android_library {
332 name: "baz",
333 sdk_version: "system_current",
334 srcs: ["c.java"],
335 }
336 `)
337
Steven Moreland00298982020-11-17 21:44:36 +0000338 testJavaError(t, "consider adjusting sdk_version: OR platform_apis:", `
Jeongik Chae403e9e2019-12-07 00:16:24 +0900339 android_app {
340 name: "foo",
341 srcs: ["a.java"],
342 libs: ["bar"],
343 sdk_version: "system_current",
344 static_libs: ["baz"],
345 }
346
347 java_library {
348 name: "bar",
349 sdk_version: "current",
350 srcs: ["b.java"],
351 }
352
353 android_library {
354 name: "baz",
355 srcs: ["c.java"],
356 }
357 `)
358}
359
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100360func TestUpdatableApps(t *testing.T) {
361 testCases := []struct {
362 name string
363 bp string
364 expectedError string
365 }{
366 {
367 name: "Stable public SDK",
368 bp: `android_app {
369 name: "foo",
370 srcs: ["a.java"],
371 sdk_version: "29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100372 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100373 updatable: true,
374 }`,
375 },
376 {
377 name: "Stable system SDK",
378 bp: `android_app {
379 name: "foo",
380 srcs: ["a.java"],
381 sdk_version: "system_29",
Artur Satayevf40fc852020-04-16 13:43:02 +0100382 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100383 updatable: true,
384 }`,
385 },
386 {
387 name: "Current public SDK",
388 bp: `android_app {
389 name: "foo",
390 srcs: ["a.java"],
391 sdk_version: "current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100392 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100393 updatable: true,
394 }`,
395 },
396 {
397 name: "Current system SDK",
398 bp: `android_app {
399 name: "foo",
400 srcs: ["a.java"],
401 sdk_version: "system_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100402 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100403 updatable: true,
404 }`,
405 },
406 {
407 name: "Current module SDK",
408 bp: `android_app {
409 name: "foo",
410 srcs: ["a.java"],
411 sdk_version: "module_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100412 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100413 updatable: true,
414 }`,
415 },
416 {
417 name: "Current core SDK",
418 bp: `android_app {
419 name: "foo",
420 srcs: ["a.java"],
421 sdk_version: "core_current",
Artur Satayevf40fc852020-04-16 13:43:02 +0100422 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100423 updatable: true,
424 }`,
425 },
426 {
427 name: "No Platform APIs",
428 bp: `android_app {
429 name: "foo",
430 srcs: ["a.java"],
431 platform_apis: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100432 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100433 updatable: true,
434 }`,
435 expectedError: "Updatable apps must use stable SDKs",
436 },
437 {
438 name: "No Core Platform APIs",
439 bp: `android_app {
440 name: "foo",
441 srcs: ["a.java"],
442 sdk_version: "core_platform",
Artur Satayevf40fc852020-04-16 13:43:02 +0100443 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100444 updatable: true,
445 }`,
446 expectedError: "Updatable apps must use stable SDKs",
447 },
448 {
449 name: "No unspecified APIs",
450 bp: `android_app {
451 name: "foo",
452 srcs: ["a.java"],
453 updatable: true,
Artur Satayevf40fc852020-04-16 13:43:02 +0100454 min_sdk_version: "29",
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100455 }`,
456 expectedError: "Updatable apps must use stable SDK",
457 },
Artur Satayevf40fc852020-04-16 13:43:02 +0100458 {
459 name: "Must specify min_sdk_version",
460 bp: `android_app {
461 name: "app_without_min_sdk_version",
462 srcs: ["a.java"],
463 sdk_version: "29",
464 updatable: true,
465 }`,
466 expectedError: "updatable apps must set min_sdk_version.",
467 },
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100468 }
469
470 for _, test := range testCases {
471 t.Run(test.name, func(t *testing.T) {
472 if test.expectedError == "" {
473 testJava(t, test.bp)
474 } else {
475 testJavaError(t, test.expectedError, test.bp)
476 }
477 })
478 }
479}
480
Jooyung Han749dc692020-04-15 11:03:39 +0900481func TestUpdatableApps_TransitiveDepsShouldSetMinSdkVersion(t *testing.T) {
482 testJavaError(t, `module "bar".*: should support min_sdk_version\(29\)`, cc.GatherRequiredDepsForTest(android.Android)+`
483 android_app {
484 name: "foo",
485 srcs: ["a.java"],
486 updatable: true,
487 sdk_version: "current",
488 min_sdk_version: "29",
489 static_libs: ["bar"],
490 }
491
492 java_library {
493 name: "bar",
494 sdk_version: "current",
495 }
496 `)
497}
498
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900499func TestUpdatableApps_JniLibsShouldShouldSupportMinSdkVersion(t *testing.T) {
500 testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
501 android_app {
502 name: "foo",
503 srcs: ["a.java"],
504 updatable: true,
505 sdk_version: "current",
506 min_sdk_version: "current",
507 jni_libs: ["libjni"],
508 }
509
510 cc_library {
511 name: "libjni",
512 stl: "none",
513 system_shared_libs: [],
514 sdk_version: "current",
515 }
516 `)
517}
518
519func TestUpdatableApps_JniLibShouldBeBuiltAgainstMinSdkVersion(t *testing.T) {
520 bp := cc.GatherRequiredDepsForTest(android.Android) + `
521 android_app {
522 name: "foo",
523 srcs: ["a.java"],
524 updatable: true,
525 sdk_version: "current",
526 min_sdk_version: "29",
527 jni_libs: ["libjni"],
528 }
529
530 cc_library {
531 name: "libjni",
532 stl: "none",
533 system_shared_libs: [],
534 sdk_version: "29",
535 }
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900536 `
537 fs := map[string][]byte{
538 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtbegin_so.o": nil,
539 "prebuilts/ndk/current/platforms/android-29/arch-arm64/usr/lib/crtend_so.o": nil,
540 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtbegin_so.o": nil,
541 "prebuilts/ndk/current/platforms/android-29/arch-arm/usr/lib/crtend_so.o": nil,
542 }
543
544 ctx, _ := testJavaWithConfig(t, testConfig(nil, bp, fs))
545
546 inputs := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").Description("link").Implicits
547 var crtbeginFound, crtendFound bool
Dan Albert92fe7402020-07-15 13:33:30 -0700548 expectedCrtBegin := ctx.ModuleForTests("crtbegin_so",
549 "android_arm64_armv8-a_sdk_29").Rule("partialLd").Output
550 expectedCrtEnd := ctx.ModuleForTests("crtend_so",
551 "android_arm64_armv8-a_sdk_29").Rule("partialLd").Output
552 implicits := []string{}
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900553 for _, input := range inputs {
Dan Albert92fe7402020-07-15 13:33:30 -0700554 implicits = append(implicits, input.String())
555 if strings.HasSuffix(input.String(), expectedCrtBegin.String()) {
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900556 crtbeginFound = true
Dan Albert92fe7402020-07-15 13:33:30 -0700557 } else if strings.HasSuffix(input.String(), expectedCrtEnd.String()) {
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900558 crtendFound = true
559 }
560 }
Dan Albert92fe7402020-07-15 13:33:30 -0700561 if !crtbeginFound {
562 t.Error(fmt.Sprintf(
563 "expected implicit with suffix %q, have the following implicits:\n%s",
564 expectedCrtBegin, strings.Join(implicits, "\n")))
565 }
566 if !crtendFound {
567 t.Error(fmt.Sprintf(
568 "expected implicit with suffix %q, have the following implicits:\n%s",
569 expectedCrtEnd, strings.Join(implicits, "\n")))
Jooyung Hanbbc3fb72020-04-29 14:01:06 +0900570 }
571}
572
573func TestUpdatableApps_ErrorIfJniLibDoesntSupportMinSdkVersion(t *testing.T) {
574 bp := cc.GatherRequiredDepsForTest(android.Android) + `
575 android_app {
576 name: "foo",
577 srcs: ["a.java"],
578 updatable: true,
579 sdk_version: "current",
580 min_sdk_version: "29", // this APK should support 29
581 jni_libs: ["libjni"],
582 }
583
584 cc_library {
585 name: "libjni",
586 stl: "none",
587 sdk_version: "current",
588 }
589 `
590 testJavaError(t, `"libjni" .*: sdk_version\(current\) is higher than min_sdk_version\(29\)`, bp)
591}
592
593func TestUpdatableApps_ErrorIfDepSdkVersionIsHigher(t *testing.T) {
594 bp := cc.GatherRequiredDepsForTest(android.Android) + `
595 android_app {
596 name: "foo",
597 srcs: ["a.java"],
598 updatable: true,
599 sdk_version: "current",
600 min_sdk_version: "29", // this APK should support 29
601 jni_libs: ["libjni"],
602 }
603
604 cc_library {
605 name: "libjni",
606 stl: "none",
607 shared_libs: ["libbar"],
608 system_shared_libs: [],
609 sdk_version: "27",
610 }
611
612 cc_library {
613 name: "libbar",
614 stl: "none",
615 system_shared_libs: [],
616 sdk_version: "current",
617 }
618 `
619 testJavaError(t, `"libjni" .*: links "libbar" built against newer API version "current"`, bp)
620}
621
Colin Cross0ddae7f2019-02-07 15:30:01 -0800622func TestResourceDirs(t *testing.T) {
623 testCases := []struct {
624 name string
625 prop string
626 resources []string
627 }{
628 {
629 name: "no resource_dirs",
630 prop: "",
631 resources: []string{"res/res/values/strings.xml"},
632 },
633 {
634 name: "resource_dirs",
635 prop: `resource_dirs: ["res"]`,
636 resources: []string{"res/res/values/strings.xml"},
637 },
638 {
639 name: "empty resource_dirs",
640 prop: `resource_dirs: []`,
641 resources: nil,
642 },
643 }
644
645 fs := map[string][]byte{
646 "res/res/values/strings.xml": nil,
647 }
648
649 bp := `
650 android_app {
651 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900652 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800653 %s
654 }
655 `
656
657 for _, testCase := range testCases {
658 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800659 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
Colin Crossae8600b2020-10-29 17:09:13 -0700660 ctx := testContext(config)
Colin Cross0ddae7f2019-02-07 15:30:01 -0800661 run(t, ctx, config)
662
663 module := ctx.ModuleForTests("foo", "android_common")
664 resourceList := module.MaybeOutput("aapt2/res.list")
665
666 var resources []string
667 if resourceList.Rule != nil {
668 for _, compiledResource := range resourceList.Inputs.Strings() {
669 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
670 }
671 }
672
673 if !reflect.DeepEqual(resources, testCase.resources) {
674 t.Errorf("expected resource files %q, got %q",
675 testCase.resources, resources)
676 }
677 })
678 }
679}
680
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800681func TestLibraryAssets(t *testing.T) {
682 bp := `
683 android_app {
684 name: "foo",
685 sdk_version: "current",
686 static_libs: ["lib1", "lib2", "lib3"],
687 }
688
689 android_library {
690 name: "lib1",
691 sdk_version: "current",
692 asset_dirs: ["assets_a"],
693 }
694
695 android_library {
696 name: "lib2",
697 sdk_version: "current",
698 }
699
700 android_library {
701 name: "lib3",
702 sdk_version: "current",
703 static_libs: ["lib4"],
704 }
705
706 android_library {
707 name: "lib4",
708 sdk_version: "current",
709 asset_dirs: ["assets_b"],
710 }
711 `
712
713 testCases := []struct {
714 name string
715 assetFlag string
716 assetPackages []string
717 }{
718 {
719 name: "foo",
720 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
721 assetPackages: []string{
722 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
723 buildDir + "/.intermediates/lib1/android_common/assets.zip",
724 buildDir + "/.intermediates/lib3/android_common/assets.zip",
725 },
726 },
727 {
728 name: "lib1",
729 assetFlag: "-A assets_a",
730 },
731 {
732 name: "lib2",
733 },
734 {
735 name: "lib3",
736 assetPackages: []string{
737 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
738 buildDir + "/.intermediates/lib4/android_common/assets.zip",
739 },
740 },
741 {
742 name: "lib4",
743 assetFlag: "-A assets_b",
744 },
745 }
746 ctx := testApp(t, bp)
747
748 for _, test := range testCases {
749 t.Run(test.name, func(t *testing.T) {
750 m := ctx.ModuleForTests(test.name, "android_common")
751
752 // Check asset flag in aapt2 link flags
753 var aapt2link android.TestingBuildParams
754 if len(test.assetPackages) > 0 {
755 aapt2link = m.Output("aapt2/package-res.apk")
756 } else {
757 aapt2link = m.Output("package-res.apk")
758 }
759 aapt2Flags := aapt2link.Args["flags"]
760 if test.assetFlag != "" {
761 if !strings.Contains(aapt2Flags, test.assetFlag) {
762 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
763 }
764 } else {
765 if strings.Contains(aapt2Flags, " -A ") {
766 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
767 }
768 }
769
770 // Check asset merge rule.
771 if len(test.assetPackages) > 0 {
772 mergeAssets := m.Output("package-res.apk")
773 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
774 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
775 mergeAssets.Inputs.Strings(), test.assetPackages)
776 }
777 }
778 })
779 }
780}
781
Colin Crossbec85302019-02-13 13:15:46 -0800782func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800783 testCases := []struct {
784 name string
785 enforceRROTargets []string
786 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800787 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800788 overlayFiles map[string][]string
789 rroDirs map[string][]string
790 }{
791 {
792 name: "no RRO",
793 enforceRROTargets: nil,
794 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800795 resourceFiles: map[string][]string{
796 "foo": nil,
797 "bar": {"bar/res/res/values/strings.xml"},
798 "lib": nil,
799 "lib2": {"lib2/res/res/values/strings.xml"},
800 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800801 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800802 "foo": {
803 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800804 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000805 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800806 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800807 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
808 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000809 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800810 },
Colin Crossbec85302019-02-13 13:15:46 -0800811 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800812 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
813 "device/vendor/blah/overlay/bar/res/values/strings.xml",
814 },
Colin Crossbec85302019-02-13 13:15:46 -0800815 "lib": {
816 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
817 "lib/res/res/values/strings.xml",
818 "device/vendor/blah/overlay/lib/res/values/strings.xml",
819 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800820 },
821 rroDirs: map[string][]string{
822 "foo": nil,
823 "bar": nil,
824 },
825 },
826 {
827 name: "enforce RRO on foo",
828 enforceRROTargets: []string{"foo"},
829 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800830 resourceFiles: map[string][]string{
831 "foo": nil,
832 "bar": {"bar/res/res/values/strings.xml"},
833 "lib": nil,
834 "lib2": {"lib2/res/res/values/strings.xml"},
835 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800836 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800837 "foo": {
838 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800839 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000840 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800841 "foo/res/res/values/strings.xml",
842 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
843 },
Colin Crossbec85302019-02-13 13:15:46 -0800844 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800845 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
846 "device/vendor/blah/overlay/bar/res/values/strings.xml",
847 },
Colin Crossbec85302019-02-13 13:15:46 -0800848 "lib": {
849 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
850 "lib/res/res/values/strings.xml",
Colin Crossbec85302019-02-13 13:15:46 -0800851 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800852 },
Colin Crossc1c37552019-01-31 11:42:41 -0800853
Colin Cross5c4791c2019-02-01 11:44:44 -0800854 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800855 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000856 "device:device/vendor/blah/overlay/foo/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000857 "product:product/vendor/blah/overlay/foo/res",
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700858 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800859 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800860 "bar": nil,
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700861 "lib": {"device:device/vendor/blah/overlay/lib/res"},
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 Crossae8600b2020-10-29 17:09:13 -0700976 ctx := testContext(config)
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) {
Colin Crossae8600b2020-10-29 17:09:13 -07001042 ctx := testContext(config)
Jeongik Cha219141c2020-08-06 23:00:37 +09001043
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
Dan Albert4f378d72020-07-23 17:32:15 -07001079 activeCodenames []string
Colin Crossd09b0b62018-04-18 11:06:47 -07001080 }{
1081 {
1082 name: "current final SDK",
1083 sdkVersion: "current",
1084 platformSdkInt: 27,
1085 platformSdkCodename: "REL",
1086 platformSdkFinal: true,
1087 expectedMinSdkVersion: "27",
1088 },
1089 {
1090 name: "current non-final SDK",
1091 sdkVersion: "current",
1092 platformSdkInt: 27,
1093 platformSdkCodename: "OMR1",
1094 platformSdkFinal: false,
1095 expectedMinSdkVersion: "OMR1",
Dan Albert4f378d72020-07-23 17:32:15 -07001096 activeCodenames: []string{"OMR1"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001097 },
1098 {
1099 name: "default final SDK",
1100 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001101 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001102 platformSdkInt: 27,
1103 platformSdkCodename: "REL",
1104 platformSdkFinal: true,
1105 expectedMinSdkVersion: "27",
1106 },
1107 {
1108 name: "default non-final SDK",
1109 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001110 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -07001111 platformSdkInt: 27,
1112 platformSdkCodename: "OMR1",
1113 platformSdkFinal: false,
1114 expectedMinSdkVersion: "OMR1",
Dan Albert4f378d72020-07-23 17:32:15 -07001115 activeCodenames: []string{"OMR1"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001116 },
1117 {
1118 name: "14",
1119 sdkVersion: "14",
1120 expectedMinSdkVersion: "14",
Dan Albert4f378d72020-07-23 17:32:15 -07001121 platformSdkCodename: "S",
1122 activeCodenames: []string{"S"},
Colin Crossd09b0b62018-04-18 11:06:47 -07001123 },
1124 }
1125
1126 for _, moduleType := range []string{"android_app", "android_library"} {
1127 for _, test := range testCases {
1128 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +09001129 platformApiProp := ""
1130 if test.platformApis {
1131 platformApiProp = "platform_apis: true,"
1132 }
Colin Crossd09b0b62018-04-18 11:06:47 -07001133 bp := fmt.Sprintf(`%s {
1134 name: "foo",
1135 srcs: ["a.java"],
1136 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001137 %s
1138 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -07001139
Colin Cross98be1bb2019-12-13 20:41:13 -08001140 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -07001141 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1142 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
Dan Albert4f378d72020-07-23 17:32:15 -07001143 config.TestProductVariables.Platform_version_active_codenames = test.activeCodenames
Colin Crossd09b0b62018-04-18 11:06:47 -07001144 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
Jeongik Cha219141c2020-08-06 23:00:37 +09001145 checkSdkVersion(t, config, test.expectedMinSdkVersion)
Colin Crossd09b0b62018-04-18 11:06:47 -07001146
Colin Crossd09b0b62018-04-18 11:06:47 -07001147 })
1148 }
1149 }
1150}
Colin Crossa4f08812018-10-02 22:03:40 -07001151
Jeongik Cha219141c2020-08-06 23:00:37 +09001152func TestVendorAppSdkVersion(t *testing.T) {
1153 testCases := []struct {
1154 name string
1155 sdkVersion string
1156 platformSdkInt int
1157 platformSdkCodename string
1158 platformSdkFinal bool
1159 deviceCurrentApiLevelForVendorModules string
1160 expectedMinSdkVersion string
1161 }{
1162 {
1163 name: "current final SDK",
1164 sdkVersion: "current",
1165 platformSdkInt: 29,
1166 platformSdkCodename: "REL",
1167 platformSdkFinal: true,
1168 deviceCurrentApiLevelForVendorModules: "29",
1169 expectedMinSdkVersion: "29",
1170 },
1171 {
1172 name: "current final SDK",
1173 sdkVersion: "current",
1174 platformSdkInt: 29,
1175 platformSdkCodename: "REL",
1176 platformSdkFinal: true,
1177 deviceCurrentApiLevelForVendorModules: "28",
1178 expectedMinSdkVersion: "28",
1179 },
1180 {
1181 name: "current final SDK",
1182 sdkVersion: "current",
1183 platformSdkInt: 29,
1184 platformSdkCodename: "Q",
1185 platformSdkFinal: false,
Jeongik Cha219141c2020-08-06 23:00:37 +09001186 deviceCurrentApiLevelForVendorModules: "28",
1187 expectedMinSdkVersion: "28",
1188 },
1189 }
1190
1191 for _, moduleType := range []string{"android_app", "android_library"} {
1192 for _, sdkKind := range []string{"", "system_"} {
1193 for _, test := range testCases {
1194 t.Run(moduleType+" "+test.name, func(t *testing.T) {
1195 bp := fmt.Sprintf(`%s {
1196 name: "foo",
1197 srcs: ["a.java"],
1198 sdk_version: "%s%s",
1199 vendor: true,
1200 }`, moduleType, sdkKind, test.sdkVersion)
1201
1202 config := testAppConfig(nil, bp, nil)
1203 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
1204 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
1205 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
1206 config.TestProductVariables.DeviceCurrentApiLevelForVendorModules = &test.deviceCurrentApiLevelForVendorModules
1207 config.TestProductVariables.DeviceSystemSdkVersions = []string{"28", "29"}
1208 checkSdkVersion(t, config, test.expectedMinSdkVersion)
1209 })
1210 }
1211 }
1212 }
1213}
1214
Paul Duffin50c217c2019-06-12 13:25:22 +01001215func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001216 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001217 cc_library {
1218 name: "libjni",
1219 system_shared_libs: [],
Colin Crossc511bc52020-04-07 16:50:32 +00001220 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001221 stl: "none",
1222 }
1223
1224 android_test {
1225 name: "test",
1226 sdk_version: "core_platform",
1227 jni_libs: ["libjni"],
1228 }
1229
1230 android_test {
1231 name: "test_first",
1232 sdk_version: "core_platform",
1233 compile_multilib: "first",
1234 jni_libs: ["libjni"],
1235 }
1236
1237 android_test {
1238 name: "test_both",
1239 sdk_version: "core_platform",
1240 compile_multilib: "both",
1241 jni_libs: ["libjni"],
1242 }
1243
1244 android_test {
1245 name: "test_32",
1246 sdk_version: "core_platform",
1247 compile_multilib: "32",
1248 jni_libs: ["libjni"],
1249 }
1250
1251 android_test {
1252 name: "test_64",
1253 sdk_version: "core_platform",
1254 compile_multilib: "64",
1255 jni_libs: ["libjni"],
1256 }
1257 `)
1258
1259 testCases := []struct {
1260 name string
1261 abis []string
1262 }{
1263 {"test", []string{"arm64-v8a"}},
1264 {"test_first", []string{"arm64-v8a"}},
1265 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
1266 {"test_32", []string{"armeabi-v7a"}},
1267 {"test_64", []string{"arm64-v8a"}},
1268 }
1269
1270 for _, test := range testCases {
1271 t.Run(test.name, func(t *testing.T) {
1272 app := ctx.ModuleForTests(test.name, "android_common")
1273 jniLibZip := app.Output("jnilibs.zip")
1274 var abis []string
1275 args := strings.Fields(jniLibZip.Args["jarArgs"])
1276 for i := 0; i < len(args); i++ {
1277 if args[i] == "-P" {
1278 abis = append(abis, filepath.Base(args[i+1]))
1279 i++
1280 }
1281 }
1282 if !reflect.DeepEqual(abis, test.abis) {
1283 t.Errorf("want abis %v, got %v", test.abis, abis)
1284 }
1285 })
1286 }
1287}
1288
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001289func TestAppSdkVersionByPartition(t *testing.T) {
1290 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
1291 android_app {
1292 name: "foo",
1293 srcs: ["a.java"],
1294 vendor: true,
1295 platform_apis: true,
1296 }
1297 `)
1298
1299 testJava(t, `
1300 android_app {
1301 name: "bar",
1302 srcs: ["b.java"],
1303 platform_apis: true,
1304 }
1305 `)
1306
1307 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001308 bp := `
1309 android_app {
1310 name: "foo",
1311 srcs: ["a.java"],
1312 product_specific: true,
1313 platform_apis: true,
1314 }
1315 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001316
1317 config := testAppConfig(nil, bp, nil)
1318 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001319 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001320 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 +09001321 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001322 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001323 }
1324 }
1325}
1326
Paul Duffin50c217c2019-06-12 13:25:22 +01001327func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001328 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001329 cc_library {
1330 name: "libjni",
1331 system_shared_libs: [],
1332 stl: "none",
Colin Cross094cde42020-02-15 10:38:00 -08001333 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001334 }
1335
1336 android_app {
1337 name: "app",
1338 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001339 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001340 }
1341
1342 android_app {
1343 name: "app_noembed",
1344 jni_libs: ["libjni"],
1345 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001346 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001347 }
1348
1349 android_app {
1350 name: "app_embed",
1351 jni_libs: ["libjni"],
1352 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001353 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001354 }
1355
1356 android_test {
1357 name: "test",
Colin Crossc511bc52020-04-07 16:50:32 +00001358 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001359 jni_libs: ["libjni"],
1360 }
1361
1362 android_test {
1363 name: "test_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001364 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001365 jni_libs: ["libjni"],
1366 use_embedded_native_libs: false,
1367 }
1368
1369 android_test_helper_app {
1370 name: "test_helper",
Colin Crossc511bc52020-04-07 16:50:32 +00001371 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001372 jni_libs: ["libjni"],
1373 }
1374
1375 android_test_helper_app {
1376 name: "test_helper_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +00001377 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001378 jni_libs: ["libjni"],
1379 use_embedded_native_libs: false,
1380 }
1381 `)
1382
1383 testCases := []struct {
1384 name string
1385 packaged bool
1386 compressed bool
1387 }{
1388 {"app", false, false},
1389 {"app_noembed", false, false},
1390 {"app_embed", true, false},
1391 {"test", true, false},
1392 {"test_noembed", true, true},
1393 {"test_helper", true, false},
1394 {"test_helper_noembed", true, true},
1395 }
1396
1397 for _, test := range testCases {
1398 t.Run(test.name, func(t *testing.T) {
1399 app := ctx.ModuleForTests(test.name, "android_common")
1400 jniLibZip := app.MaybeOutput("jnilibs.zip")
1401 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1402 t.Errorf("expected jni packaged %v, got %v", w, g)
1403 }
1404
1405 if jniLibZip.Rule != nil {
1406 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1407 t.Errorf("expected jni compressed %v, got %v", w, g)
1408 }
Colin Crossc511bc52020-04-07 16:50:32 +00001409
1410 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1411 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1412 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001413 }
1414 })
1415 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001416}
1417
Colin Cross3c007702020-05-08 11:20:24 -07001418func TestJNISDK(t *testing.T) {
1419 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1420 cc_library {
1421 name: "libjni",
1422 system_shared_libs: [],
1423 stl: "none",
1424 sdk_version: "current",
1425 }
1426
1427 android_test {
1428 name: "app_platform",
1429 jni_libs: ["libjni"],
1430 platform_apis: true,
1431 }
1432
1433 android_test {
1434 name: "app_sdk",
1435 jni_libs: ["libjni"],
1436 sdk_version: "current",
1437 }
1438
1439 android_test {
1440 name: "app_force_platform",
1441 jni_libs: ["libjni"],
1442 sdk_version: "current",
1443 jni_uses_platform_apis: true,
1444 }
1445
1446 android_test {
1447 name: "app_force_sdk",
1448 jni_libs: ["libjni"],
1449 platform_apis: true,
1450 jni_uses_sdk_apis: true,
1451 }
Colin Crossc2d24052020-05-13 11:05:02 -07001452
1453 cc_library {
1454 name: "libvendorjni",
1455 system_shared_libs: [],
1456 stl: "none",
1457 vendor: true,
1458 }
1459
1460 android_test {
1461 name: "app_vendor",
1462 jni_libs: ["libvendorjni"],
1463 sdk_version: "current",
1464 vendor: true,
1465 }
Colin Cross3c007702020-05-08 11:20:24 -07001466 `)
1467
1468 testCases := []struct {
Colin Crossc2d24052020-05-13 11:05:02 -07001469 name string
1470 sdkJNI bool
1471 vendorJNI bool
Colin Cross3c007702020-05-08 11:20:24 -07001472 }{
Colin Crossc2d24052020-05-13 11:05:02 -07001473 {name: "app_platform"},
1474 {name: "app_sdk", sdkJNI: true},
1475 {name: "app_force_platform"},
1476 {name: "app_force_sdk", sdkJNI: true},
1477 {name: "app_vendor", vendorJNI: true},
Colin Cross3c007702020-05-08 11:20:24 -07001478 }
1479
Colin Crossc2d24052020-05-13 11:05:02 -07001480 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1481 Output("libjni.so").Output.String()
1482 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1483 Output("libjni.so").Output.String()
1484 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1485 Output("libvendorjni.so").Output.String()
1486
Colin Cross3c007702020-05-08 11:20:24 -07001487 for _, test := range testCases {
1488 t.Run(test.name, func(t *testing.T) {
1489 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross3c007702020-05-08 11:20:24 -07001490
1491 jniLibZip := app.MaybeOutput("jnilibs.zip")
1492 if len(jniLibZip.Implicits) != 1 {
1493 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1494 }
1495 gotJNI := jniLibZip.Implicits[0].String()
1496
1497 if test.sdkJNI {
1498 if gotJNI != sdkJNI {
1499 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1500 }
Colin Crossc2d24052020-05-13 11:05:02 -07001501 } else if test.vendorJNI {
1502 if gotJNI != vendorJNI {
1503 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1504 }
Colin Cross3c007702020-05-08 11:20:24 -07001505 } else {
1506 if gotJNI != platformJNI {
1507 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1508 }
1509 }
1510 })
1511 }
1512
1513 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1514 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1515 android_test {
1516 name: "app_platform",
1517 platform_apis: true,
1518 jni_uses_platform_apis: true,
1519 }
1520 `)
1521 })
1522
1523 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1524 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1525 android_test {
1526 name: "app_sdk",
1527 sdk_version: "current",
1528 jni_uses_sdk_apis: true,
1529 }
1530 `)
1531 })
1532
1533}
1534
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001535func TestCertificates(t *testing.T) {
1536 testCases := []struct {
1537 name string
1538 bp string
1539 certificateOverride string
Liz Kammere2b27f42020-05-07 13:24:05 -07001540 expectedLineage string
1541 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001542 }{
1543 {
1544 name: "default",
1545 bp: `
1546 android_app {
1547 name: "foo",
1548 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001549 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001550 }
1551 `,
1552 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001553 expectedLineage: "",
1554 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001555 },
1556 {
1557 name: "module certificate property",
1558 bp: `
1559 android_app {
1560 name: "foo",
1561 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001562 certificate: ":new_certificate",
1563 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001564 }
1565
1566 android_app_certificate {
1567 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001568 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001569 }
1570 `,
1571 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001572 expectedLineage: "",
1573 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001574 },
1575 {
1576 name: "path certificate property",
1577 bp: `
1578 android_app {
1579 name: "foo",
1580 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001581 certificate: "expiredkey",
1582 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001583 }
1584 `,
1585 certificateOverride: "",
Liz Kammere2b27f42020-05-07 13:24:05 -07001586 expectedLineage: "",
1587 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001588 },
1589 {
1590 name: "certificate overrides",
1591 bp: `
1592 android_app {
1593 name: "foo",
1594 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001595 certificate: "expiredkey",
1596 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001597 }
1598
1599 android_app_certificate {
1600 name: "new_certificate",
Colin Cross3c007702020-05-08 11:20:24 -07001601 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001602 }
1603 `,
1604 certificateOverride: "foo:new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001605 expectedLineage: "",
1606 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1607 },
1608 {
1609 name: "certificate lineage",
1610 bp: `
1611 android_app {
1612 name: "foo",
1613 srcs: ["a.java"],
1614 certificate: ":new_certificate",
1615 lineage: "lineage.bin",
1616 sdk_version: "current",
1617 }
1618
1619 android_app_certificate {
1620 name: "new_certificate",
1621 certificate: "cert/new_cert",
1622 }
1623 `,
1624 certificateOverride: "",
1625 expectedLineage: "--lineage lineage.bin",
1626 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001627 },
1628 }
1629
1630 for _, test := range testCases {
1631 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001632 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001633 if test.certificateOverride != "" {
1634 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1635 }
Colin Crossae8600b2020-10-29 17:09:13 -07001636 ctx := testContext(config)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001637
1638 run(t, ctx, config)
1639 foo := ctx.ModuleForTests("foo", "android_common")
1640
1641 signapk := foo.Output("foo.apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001642 signCertificateFlags := signapk.Args["certificates"]
1643 if test.expectedCertificate != signCertificateFlags {
1644 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1645 }
1646
1647 signFlags := signapk.Args["flags"]
1648 if test.expectedLineage != signFlags {
1649 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001650 }
1651 })
1652 }
1653}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001654
Songchun Fan688de9a2020-03-24 20:32:24 -07001655func TestRequestV4SigningFlag(t *testing.T) {
1656 testCases := []struct {
1657 name string
1658 bp string
1659 expected string
1660 }{
1661 {
1662 name: "default",
1663 bp: `
1664 android_app {
1665 name: "foo",
1666 srcs: ["a.java"],
1667 sdk_version: "current",
1668 }
1669 `,
1670 expected: "",
1671 },
1672 {
1673 name: "default",
1674 bp: `
1675 android_app {
1676 name: "foo",
1677 srcs: ["a.java"],
1678 sdk_version: "current",
1679 v4_signature: false,
1680 }
1681 `,
1682 expected: "",
1683 },
1684 {
1685 name: "module certificate property",
1686 bp: `
1687 android_app {
1688 name: "foo",
1689 srcs: ["a.java"],
1690 sdk_version: "current",
1691 v4_signature: true,
1692 }
1693 `,
1694 expected: "--enable-v4",
1695 },
1696 }
1697
1698 for _, test := range testCases {
1699 t.Run(test.name, func(t *testing.T) {
1700 config := testAppConfig(nil, test.bp, nil)
Colin Crossae8600b2020-10-29 17:09:13 -07001701 ctx := testContext(config)
Songchun Fan688de9a2020-03-24 20:32:24 -07001702
1703 run(t, ctx, config)
1704 foo := ctx.ModuleForTests("foo", "android_common")
1705
1706 signapk := foo.Output("foo.apk")
1707 signFlags := signapk.Args["flags"]
1708 if test.expected != signFlags {
1709 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1710 }
1711 })
1712 }
1713}
1714
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001715func TestPackageNameOverride(t *testing.T) {
1716 testCases := []struct {
1717 name string
1718 bp string
1719 packageNameOverride string
1720 expected []string
1721 }{
1722 {
1723 name: "default",
1724 bp: `
1725 android_app {
1726 name: "foo",
1727 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001728 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001729 }
1730 `,
1731 packageNameOverride: "",
1732 expected: []string{
1733 buildDir + "/.intermediates/foo/android_common/foo.apk",
1734 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1735 },
1736 },
1737 {
1738 name: "overridden",
1739 bp: `
1740 android_app {
1741 name: "foo",
1742 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001743 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001744 }
1745 `,
1746 packageNameOverride: "foo:bar",
1747 expected: []string{
1748 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001749 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001750 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1751 },
1752 },
1753 }
1754
1755 for _, test := range testCases {
1756 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001757 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001758 if test.packageNameOverride != "" {
1759 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1760 }
Colin Crossae8600b2020-10-29 17:09:13 -07001761 ctx := testContext(config)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001762
1763 run(t, ctx, config)
1764 foo := ctx.ModuleForTests("foo", "android_common")
1765
1766 outputs := foo.AllOutputs()
1767 outputMap := make(map[string]bool)
1768 for _, o := range outputs {
1769 outputMap[o] = true
1770 }
1771 for _, e := range test.expected {
1772 if _, exist := outputMap[e]; !exist {
1773 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1774 }
1775 }
1776 })
1777 }
1778}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001779
1780func TestInstrumentationTargetOverridden(t *testing.T) {
1781 bp := `
1782 android_app {
1783 name: "foo",
1784 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001785 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001786 }
1787
1788 android_test {
1789 name: "bar",
1790 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001791 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001792 }
1793 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001794 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001795 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Crossae8600b2020-10-29 17:09:13 -07001796 ctx := testContext(config)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001797
1798 run(t, ctx, config)
1799
1800 bar := ctx.ModuleForTests("bar", "android_common")
1801 res := bar.Output("package-res.apk")
1802 aapt2Flags := res.Args["flags"]
1803 e := "--rename-instrumentation-target-package org.dandroid.bp"
1804 if !strings.Contains(aapt2Flags, e) {
1805 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1806 }
1807}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001808
1809func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001810 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001811 android_app {
1812 name: "foo",
1813 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001814 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001815 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001816 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001817 }
1818
1819 override_android_app {
1820 name: "bar",
1821 base: "foo",
1822 certificate: ":new_certificate",
Liz Kammere2b27f42020-05-07 13:24:05 -07001823 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001824 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001825 }
1826
1827 android_app_certificate {
1828 name: "new_certificate",
1829 certificate: "cert/new_cert",
1830 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001831
1832 override_android_app {
1833 name: "baz",
1834 base: "foo",
1835 package_name: "org.dandroid.bp",
1836 }
Liz Kammer9f9fd022020-06-18 19:44:06 +00001837
1838 override_android_app {
1839 name: "baz_no_rename_resources",
1840 base: "foo",
1841 package_name: "org.dandroid.bp",
1842 rename_resources_package: false,
1843 }
1844
1845 android_app {
1846 name: "foo_no_rename_resources",
1847 srcs: ["a.java"],
1848 certificate: "expiredkey",
1849 overrides: ["qux"],
1850 rename_resources_package: false,
1851 sdk_version: "current",
1852 }
1853
1854 override_android_app {
1855 name: "baz_base_no_rename_resources",
1856 base: "foo_no_rename_resources",
1857 package_name: "org.dandroid.bp",
1858 }
1859
1860 override_android_app {
1861 name: "baz_override_base_rename_resources",
1862 base: "foo_no_rename_resources",
1863 package_name: "org.dandroid.bp",
1864 rename_resources_package: true,
1865 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001866 `)
1867
1868 expectedVariants := []struct {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001869 name string
1870 moduleName string
1871 variantName string
1872 apkName string
1873 apkPath string
1874 certFlag string
1875 lineageFlag string
1876 overrides []string
1877 packageFlag string
1878 renameResources bool
1879 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001880 }{
1881 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001882 name: "foo",
1883 moduleName: "foo",
1884 variantName: "android_common",
1885 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
1886 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1887 lineageFlag: "",
1888 overrides: []string{"qux"},
1889 packageFlag: "",
1890 renameResources: false,
1891 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001892 },
1893 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001894 name: "foo",
1895 moduleName: "bar",
1896 variantName: "android_common_bar",
1897 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
1898 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1899 lineageFlag: "--lineage lineage.bin",
1900 overrides: []string{"qux", "foo"},
1901 packageFlag: "",
1902 renameResources: false,
1903 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001904 },
1905 {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001906 name: "foo",
1907 moduleName: "baz",
1908 variantName: "android_common_baz",
1909 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
1910 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1911 lineageFlag: "",
1912 overrides: []string{"qux", "foo"},
1913 packageFlag: "org.dandroid.bp",
1914 renameResources: true,
1915 logging_parent: "",
1916 },
1917 {
1918 name: "foo",
1919 moduleName: "baz_no_rename_resources",
1920 variantName: "android_common_baz_no_rename_resources",
1921 apkPath: "/target/product/test_device/system/app/baz_no_rename_resources/baz_no_rename_resources.apk",
1922 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1923 lineageFlag: "",
1924 overrides: []string{"qux", "foo"},
1925 packageFlag: "org.dandroid.bp",
1926 renameResources: false,
1927 logging_parent: "",
1928 },
1929 {
1930 name: "foo_no_rename_resources",
1931 moduleName: "baz_base_no_rename_resources",
1932 variantName: "android_common_baz_base_no_rename_resources",
1933 apkPath: "/target/product/test_device/system/app/baz_base_no_rename_resources/baz_base_no_rename_resources.apk",
1934 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1935 lineageFlag: "",
1936 overrides: []string{"qux", "foo_no_rename_resources"},
1937 packageFlag: "org.dandroid.bp",
1938 renameResources: false,
1939 logging_parent: "",
1940 },
1941 {
1942 name: "foo_no_rename_resources",
1943 moduleName: "baz_override_base_rename_resources",
1944 variantName: "android_common_baz_override_base_rename_resources",
1945 apkPath: "/target/product/test_device/system/app/baz_override_base_rename_resources/baz_override_base_rename_resources.apk",
1946 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1947 lineageFlag: "",
1948 overrides: []string{"qux", "foo_no_rename_resources"},
1949 packageFlag: "org.dandroid.bp",
1950 renameResources: true,
1951 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001952 },
1953 }
1954 for _, expected := range expectedVariants {
Liz Kammer9f9fd022020-06-18 19:44:06 +00001955 variant := ctx.ModuleForTests(expected.name, expected.variantName)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001956
1957 // Check the final apk name
1958 outputs := variant.AllOutputs()
1959 expectedApkPath := buildDir + expected.apkPath
1960 found := false
1961 for _, o := range outputs {
1962 if o == expectedApkPath {
1963 found = true
1964 break
1965 }
1966 }
1967 if !found {
1968 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1969 }
1970
1971 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001972 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammere2b27f42020-05-07 13:24:05 -07001973 certFlag := signapk.Args["certificates"]
1974 if expected.certFlag != certFlag {
1975 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1976 }
1977
1978 // Check the lineage flags
1979 lineageFlag := signapk.Args["flags"]
1980 if expected.lineageFlag != lineageFlag {
1981 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001982 }
1983
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001984 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001985 mod := variant.Module().(*AndroidApp)
1986 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1987 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1988 expected.overrides, mod.appProperties.Overrides)
1989 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001990
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001991 // Test Overridable property: Logging_parent
1992 logging_parent := mod.aapt.LoggingParent
1993 if expected.logging_parent != logging_parent {
1994 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1995 expected.logging_parent, logging_parent)
1996 }
1997
Liz Kammer1d5983b2020-05-19 19:15:37 +00001998 // Check the package renaming flag, if exists.
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001999 res := variant.Output("package-res.apk")
2000 aapt2Flags := res.Args["flags"]
Liz Kammer9f9fd022020-06-18 19:44:06 +00002001 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
2002 expectedPackage := expected.packageFlag
2003 if !expected.renameResources {
2004 expectedPackage = ""
Liz Kammer1d5983b2020-05-19 19:15:37 +00002005 }
Liz Kammer9f9fd022020-06-18 19:44:06 +00002006 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expectedPackage)
Jaewoong Jung525443a2019-02-28 15:35:54 -08002007 }
2008}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002009
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07002010func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002011 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07002012 android_app {
2013 name: "foo",
2014 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002015 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07002016 }
2017
2018 override_android_app {
2019 name: "bar",
2020 base: "foo",
2021 package_name: "org.dandroid.bp",
2022 }
2023
2024 android_test {
2025 name: "baz",
2026 srcs: ["b.java"],
2027 instrumentation_for: "foo",
2028 }
2029
2030 android_test {
2031 name: "qux",
2032 srcs: ["b.java"],
2033 instrumentation_for: "bar",
2034 }
2035 `)
2036
2037 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
2038 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
2039 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
2040 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
2041 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
2042 }
2043
2044 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
2045 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
2046 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
2047 if !strings.Contains(javac.Args["classpath"], barTurbine) {
2048 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
2049 }
2050}
2051
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002052func TestOverrideAndroidTest(t *testing.T) {
2053 ctx, _ := testJava(t, `
2054 android_app {
2055 name: "foo",
2056 srcs: ["a.java"],
2057 package_name: "com.android.foo",
2058 sdk_version: "current",
2059 }
2060
2061 override_android_app {
2062 name: "bar",
2063 base: "foo",
2064 package_name: "com.android.bar",
2065 }
2066
2067 android_test {
2068 name: "foo_test",
2069 srcs: ["b.java"],
2070 instrumentation_for: "foo",
2071 }
2072
2073 override_android_test {
2074 name: "bar_test",
2075 base: "foo_test",
2076 package_name: "com.android.bar.test",
2077 instrumentation_for: "bar",
2078 instrumentation_target_package: "com.android.bar",
2079 }
2080 `)
2081
2082 expectedVariants := []struct {
2083 moduleName string
2084 variantName string
2085 apkPath string
2086 overrides []string
2087 targetVariant string
2088 packageFlag string
2089 targetPackageFlag string
2090 }{
2091 {
2092 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002093 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002094 overrides: nil,
2095 targetVariant: "android_common",
2096 packageFlag: "",
2097 targetPackageFlag: "",
2098 },
2099 {
2100 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08002101 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002102 overrides: []string{"foo_test"},
2103 targetVariant: "android_common_bar",
2104 packageFlag: "com.android.bar.test",
2105 targetPackageFlag: "com.android.bar",
2106 },
2107 }
2108 for _, expected := range expectedVariants {
2109 variant := ctx.ModuleForTests("foo_test", expected.variantName)
2110
2111 // Check the final apk name
2112 outputs := variant.AllOutputs()
2113 expectedApkPath := buildDir + expected.apkPath
2114 found := false
2115 for _, o := range outputs {
2116 if o == expectedApkPath {
2117 found = true
2118 break
2119 }
2120 }
2121 if !found {
2122 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
2123 }
2124
2125 // Check if the overrides field values are correctly aggregated.
2126 mod := variant.Module().(*AndroidTest)
2127 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
2128 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
2129 expected.overrides, mod.appProperties.Overrides)
2130 }
2131
2132 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
2133 javac := variant.Rule("javac")
2134 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
2135 if !strings.Contains(javac.Args["classpath"], turbine) {
2136 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
2137 }
2138
2139 // Check aapt2 flags.
2140 res := variant.Output("package-res.apk")
2141 aapt2Flags := res.Args["flags"]
2142 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammer9f9fd022020-06-18 19:44:06 +00002143 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", expected.packageFlag)
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002144 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
2145 }
2146}
2147
Jaewoong Jung39982342020-01-14 10:27:18 -08002148func TestAndroidTest_FixTestConfig(t *testing.T) {
2149 ctx, _ := testJava(t, `
2150 android_app {
2151 name: "foo",
2152 srcs: ["a.java"],
2153 package_name: "com.android.foo",
2154 sdk_version: "current",
2155 }
2156
2157 android_test {
2158 name: "foo_test",
2159 srcs: ["b.java"],
2160 instrumentation_for: "foo",
2161 }
2162
2163 android_test {
2164 name: "bar_test",
2165 srcs: ["b.java"],
2166 package_name: "com.android.bar.test",
2167 instrumentation_for: "foo",
2168 }
2169
2170 override_android_test {
2171 name: "baz_test",
2172 base: "foo_test",
2173 package_name: "com.android.baz.test",
2174 }
2175 `)
2176
2177 testCases := []struct {
2178 moduleName string
2179 variantName string
2180 expectedFlags []string
2181 }{
2182 {
2183 moduleName: "foo_test",
2184 variantName: "android_common",
2185 },
2186 {
2187 moduleName: "bar_test",
2188 variantName: "android_common",
2189 expectedFlags: []string{
2190 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
2191 "--package-name com.android.bar.test",
2192 },
2193 },
2194 {
2195 moduleName: "foo_test",
2196 variantName: "android_common_baz_test",
2197 expectedFlags: []string{
2198 "--manifest " + buildDir +
2199 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
2200 "--package-name com.android.baz.test",
2201 "--test-file-name baz_test.apk",
2202 },
2203 },
2204 }
2205
2206 for _, test := range testCases {
2207 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
2208 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
2209
2210 if len(test.expectedFlags) > 0 {
2211 if params.Rule == nil {
2212 t.Errorf("test_config_fixer was expected to run, but didn't")
2213 } else {
2214 for _, flag := range test.expectedFlags {
2215 if !strings.Contains(params.RuleParams.Command, flag) {
2216 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
2217 }
2218 }
2219 }
2220 } else {
2221 if params.Rule != nil {
2222 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
2223 }
2224 }
2225
2226 }
2227}
2228
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002229func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002230 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002231 android_app_import {
2232 name: "foo",
2233 apk: "prebuilts/apk/app.apk",
2234 certificate: "platform",
2235 dex_preopt: {
2236 enabled: true,
2237 },
2238 }
2239 `)
2240
2241 variant := ctx.ModuleForTests("foo", "android_common")
2242
2243 // Check dexpreopt outputs.
2244 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2245 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2246 t.Errorf("can't find dexpreopt outputs")
2247 }
2248
2249 // Check cert signing flag.
2250 signedApk := variant.Output("signed/foo.apk")
2251 signingFlag := signedApk.Args["certificates"]
2252 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2253 if expected != signingFlag {
2254 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2255 }
2256}
2257
2258func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002259 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002260 android_app_import {
2261 name: "foo",
2262 apk: "prebuilts/apk/app.apk",
2263 certificate: "platform",
2264 dex_preopt: {
2265 enabled: false,
2266 },
2267 }
2268 `)
2269
2270 variant := ctx.ModuleForTests("foo", "android_common")
2271
2272 // Check dexpreopt outputs. They shouldn't exist.
2273 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
2274 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
2275 t.Errorf("dexpreopt shouldn't have run.")
2276 }
2277}
2278
2279func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002280 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002281 android_app_import {
2282 name: "foo",
2283 apk: "prebuilts/apk/app.apk",
2284 presigned: true,
2285 dex_preopt: {
2286 enabled: true,
2287 },
2288 }
2289 `)
2290
2291 variant := ctx.ModuleForTests("foo", "android_common")
2292
2293 // Check dexpreopt outputs.
2294 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2295 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2296 t.Errorf("can't find dexpreopt outputs")
2297 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01002298 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07002299 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
2300 t.Errorf("signing rule shouldn't be included.")
2301 }
2302 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
2303 t.Errorf("can't find aligning rule")
2304 }
2305}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002306
Liz Kammer24978992020-05-13 15:49:21 -07002307func TestAndroidAppImport_SigningLineage(t *testing.T) {
2308 ctx, _ := testJava(t, `
2309 android_app_import {
2310 name: "foo",
2311 apk: "prebuilts/apk/app.apk",
2312 certificate: "platform",
2313 lineage: "lineage.bin",
2314 }
2315 `)
2316
2317 variant := ctx.ModuleForTests("foo", "android_common")
2318
2319 // Check cert signing lineage flag.
2320 signedApk := variant.Output("signed/foo.apk")
2321 signingFlag := signedApk.Args["flags"]
2322 expected := "--lineage lineage.bin"
2323 if expected != signingFlag {
2324 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2325 }
2326}
2327
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002328func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
2329 ctx, _ := testJava(t, `
2330 android_app_import {
2331 name: "foo",
2332 apk: "prebuilts/apk/app.apk",
2333 default_dev_cert: true,
2334 dex_preopt: {
2335 enabled: true,
2336 },
2337 }
2338 `)
2339
2340 variant := ctx.ModuleForTests("foo", "android_common")
2341
2342 // Check dexpreopt outputs.
2343 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
2344 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
2345 t.Errorf("can't find dexpreopt outputs")
2346 }
2347
2348 // Check cert signing flag.
2349 signedApk := variant.Output("signed/foo.apk")
2350 signingFlag := signedApk.Args["certificates"]
2351 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
2352 if expected != signingFlag {
2353 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2354 }
2355}
2356
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002357func TestAndroidAppImport_DpiVariants(t *testing.T) {
2358 bp := `
2359 android_app_import {
2360 name: "foo",
2361 apk: "prebuilts/apk/app.apk",
2362 dpi_variants: {
2363 xhdpi: {
2364 apk: "prebuilts/apk/app_xhdpi.apk",
2365 },
2366 xxhdpi: {
2367 apk: "prebuilts/apk/app_xxhdpi.apk",
2368 },
2369 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002370 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002371 dex_preopt: {
2372 enabled: true,
2373 },
2374 }
2375 `
2376 testCases := []struct {
2377 name string
2378 aaptPreferredConfig *string
2379 aaptPrebuiltDPI []string
2380 expected string
2381 }{
2382 {
2383 name: "no preferred",
2384 aaptPreferredConfig: nil,
2385 aaptPrebuiltDPI: []string{},
2386 expected: "prebuilts/apk/app.apk",
2387 },
2388 {
2389 name: "AAPTPreferredConfig matches",
2390 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07002391 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002392 expected: "prebuilts/apk/app_xhdpi.apk",
2393 },
2394 {
2395 name: "AAPTPrebuiltDPI matches",
2396 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2397 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
2398 expected: "prebuilts/apk/app_xxhdpi.apk",
2399 },
2400 {
2401 name: "non-first AAPTPrebuiltDPI matches",
2402 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2403 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2404 expected: "prebuilts/apk/app_xhdpi.apk",
2405 },
2406 {
2407 name: "no matches",
2408 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2409 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2410 expected: "prebuilts/apk/app.apk",
2411 },
2412 }
2413
2414 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2415 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002416 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002417 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2418 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Crossae8600b2020-10-29 17:09:13 -07002419 ctx := testContext(config)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002420
2421 run(t, ctx, config)
2422
2423 variant := ctx.ModuleForTests("foo", "android_common")
2424 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2425 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2426 if len(matches) != 2 {
2427 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2428 }
2429 if test.expected != matches[1] {
2430 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2431 }
2432 }
2433}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002434
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002435func TestAndroidAppImport_Filename(t *testing.T) {
2436 ctx, config := testJava(t, `
2437 android_app_import {
2438 name: "foo",
2439 apk: "prebuilts/apk/app.apk",
2440 presigned: true,
2441 }
2442
2443 android_app_import {
2444 name: "bar",
2445 apk: "prebuilts/apk/app.apk",
2446 presigned: true,
2447 filename: "bar_sample.apk"
2448 }
2449 `)
2450
2451 testCases := []struct {
2452 name string
2453 expected string
2454 }{
2455 {
2456 name: "foo",
2457 expected: "foo.apk",
2458 },
2459 {
2460 name: "bar",
2461 expected: "bar_sample.apk",
2462 },
2463 }
2464
2465 for _, test := range testCases {
2466 variant := ctx.ModuleForTests(test.name, "android_common")
2467 if variant.MaybeOutput(test.expected).Rule == nil {
2468 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2469 }
2470
2471 a := variant.Module().(*AndroidAppImport)
2472 expectedValues := []string{test.expected}
2473 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002474 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002475 if !reflect.DeepEqual(actualValues, expectedValues) {
2476 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2477 actualValues, expectedValues)
2478 }
2479 }
2480}
2481
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002482func TestAndroidAppImport_ArchVariants(t *testing.T) {
2483 // The test config's target arch is ARM64.
2484 testCases := []struct {
2485 name string
2486 bp string
2487 expected string
2488 }{
2489 {
2490 name: "matching arch",
2491 bp: `
2492 android_app_import {
2493 name: "foo",
2494 apk: "prebuilts/apk/app.apk",
2495 arch: {
2496 arm64: {
2497 apk: "prebuilts/apk/app_arm64.apk",
2498 },
2499 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002500 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002501 dex_preopt: {
2502 enabled: true,
2503 },
2504 }
2505 `,
2506 expected: "prebuilts/apk/app_arm64.apk",
2507 },
2508 {
2509 name: "no matching arch",
2510 bp: `
2511 android_app_import {
2512 name: "foo",
2513 apk: "prebuilts/apk/app.apk",
2514 arch: {
2515 arm: {
2516 apk: "prebuilts/apk/app_arm.apk",
2517 },
2518 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002519 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002520 dex_preopt: {
2521 enabled: true,
2522 },
2523 }
2524 `,
2525 expected: "prebuilts/apk/app.apk",
2526 },
2527 }
2528
2529 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2530 for _, test := range testCases {
2531 ctx, _ := testJava(t, test.bp)
2532
2533 variant := ctx.ModuleForTests("foo", "android_common")
2534 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2535 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2536 if len(matches) != 2 {
2537 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2538 }
2539 if test.expected != matches[1] {
2540 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2541 }
2542 }
2543}
2544
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002545func TestAndroidTestImport(t *testing.T) {
2546 ctx, config := testJava(t, `
2547 android_test_import {
2548 name: "foo",
2549 apk: "prebuilts/apk/app.apk",
2550 presigned: true,
2551 data: [
2552 "testdata/data",
2553 ],
2554 }
2555 `)
2556
2557 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2558
2559 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002560 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002561 expected := []string{"tests"}
2562 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2563 if !reflect.DeepEqual(expected, actual) {
2564 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2565 }
2566 expected = []string{"testdata/data:testdata/data"}
2567 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2568 if !reflect.DeepEqual(expected, actual) {
2569 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2570 }
2571}
2572
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002573func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2574 ctx, _ := testJava(t, `
2575 android_test_import {
2576 name: "foo",
2577 apk: "prebuilts/apk/app.apk",
2578 certificate: "cert/new_cert",
2579 data: [
2580 "testdata/data",
2581 ],
2582 }
2583
2584 android_test_import {
2585 name: "foo_presigned",
2586 apk: "prebuilts/apk/app.apk",
2587 presigned: true,
2588 data: [
2589 "testdata/data",
2590 ],
2591 }
2592 `)
2593
2594 variant := ctx.ModuleForTests("foo", "android_common")
2595 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2596 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2597 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2598 }
2599
2600 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2601 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2602 if jniRule != android.Cp.String() {
2603 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2604 }
Liz Kammer3b70b3f2020-05-20 14:36:30 -07002605 if variant.MaybeOutput("zip-aligned/foo_presigned.apk").Rule == nil {
2606 t.Errorf("Presigned test apk should be aligned")
2607 }
2608}
2609
2610func TestAndroidTestImport_Preprocessed(t *testing.T) {
2611 ctx, _ := testJava(t, `
2612 android_test_import {
2613 name: "foo",
2614 apk: "prebuilts/apk/app.apk",
2615 presigned: true,
2616 preprocessed: true,
2617 }
2618
2619 android_test_import {
2620 name: "foo_cert",
2621 apk: "prebuilts/apk/app.apk",
2622 certificate: "cert/new_cert",
2623 preprocessed: true,
2624 }
2625 `)
2626
2627 testModules := []string{"foo", "foo_cert"}
2628 for _, m := range testModules {
2629 apkName := m + ".apk"
2630 variant := ctx.ModuleForTests(m, "android_common")
2631 jniRule := variant.Output("jnis-uncompressed/" + apkName).BuildParams.Rule.String()
2632 if jniRule != android.Cp.String() {
2633 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2634 }
2635
2636 // Make sure signing and aligning were skipped.
2637 if variant.MaybeOutput("signed/"+apkName).Rule != nil {
2638 t.Errorf("signing rule shouldn't be included for preprocessed.")
2639 }
2640 if variant.MaybeOutput("zip-aligned/"+apkName).Rule != nil {
2641 t.Errorf("aligning rule shouldn't be for preprocessed")
2642 }
2643 }
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002644}
2645
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002646func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002647 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002648 cc_library {
2649 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002650 sdk_version: "current",
2651 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002652 }
2653
2654 android_test {
2655 name: "stl",
2656 jni_libs: ["libjni"],
2657 compile_multilib: "both",
2658 sdk_version: "current",
2659 stl: "c++_shared",
2660 }
2661
2662 android_test {
2663 name: "system",
2664 jni_libs: ["libjni"],
2665 compile_multilib: "both",
2666 sdk_version: "current",
2667 }
2668 `)
2669
2670 testCases := []struct {
2671 name string
2672 jnis []string
2673 }{
2674 {"stl",
2675 []string{
2676 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002677 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002678 },
2679 },
2680 {"system",
2681 []string{
2682 "libjni.so",
2683 },
2684 },
2685 }
2686
2687 for _, test := range testCases {
2688 t.Run(test.name, func(t *testing.T) {
2689 app := ctx.ModuleForTests(test.name, "android_common")
2690 jniLibZip := app.Output("jnilibs.zip")
2691 var jnis []string
2692 args := strings.Fields(jniLibZip.Args["jarArgs"])
2693 for i := 0; i < len(args); i++ {
2694 if args[i] == "-f" {
2695 jnis = append(jnis, args[i+1])
2696 i += 1
2697 }
2698 }
2699 jnisJoined := strings.Join(jnis, " ")
2700 for _, jni := range test.jnis {
2701 if !strings.Contains(jnisJoined, jni) {
2702 t.Errorf("missing jni %q in %q", jni, jnis)
2703 }
2704 }
2705 })
2706 }
2707}
Colin Cross50ddcc42019-05-16 12:28:22 -07002708
2709func TestUsesLibraries(t *testing.T) {
2710 bp := `
2711 java_sdk_library {
2712 name: "foo",
2713 srcs: ["a.java"],
2714 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002715 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002716 }
2717
2718 java_sdk_library {
Paul Duffin859fe962020-05-15 10:20:31 +01002719 name: "qux",
2720 srcs: ["a.java"],
2721 api_packages: ["qux"],
2722 sdk_version: "current",
2723 }
2724
2725 java_sdk_library {
2726 name: "quuz",
2727 srcs: ["a.java"],
2728 api_packages: ["quuz"],
2729 sdk_version: "current",
2730 }
2731
2732 java_sdk_library {
Colin Cross50ddcc42019-05-16 12:28:22 -07002733 name: "bar",
2734 srcs: ["a.java"],
2735 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002736 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002737 }
2738
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002739 java_sdk_library {
2740 name: "runtime-library",
2741 srcs: ["a.java"],
2742 sdk_version: "current",
2743 }
2744
2745 java_library {
2746 name: "static-runtime-helper",
2747 srcs: ["a.java"],
2748 libs: ["runtime-library"],
2749 sdk_version: "current",
2750 }
2751
Colin Cross50ddcc42019-05-16 12:28:22 -07002752 android_app {
2753 name: "app",
2754 srcs: ["a.java"],
Paul Duffin859fe962020-05-15 10:20:31 +01002755 libs: ["qux", "quuz.stubs"],
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002756 static_libs: ["static-runtime-helper"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002757 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002758 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002759 optional_uses_libs: [
2760 "bar",
2761 "baz",
2762 ],
2763 }
2764
2765 android_app_import {
2766 name: "prebuilt",
2767 apk: "prebuilts/apk/app.apk",
2768 certificate: "platform",
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002769 uses_libs: ["foo", "android.test.runner"],
Colin Cross50ddcc42019-05-16 12:28:22 -07002770 optional_uses_libs: [
2771 "bar",
2772 "baz",
2773 ],
2774 }
2775 `
2776
Colin Cross98be1bb2019-12-13 20:41:13 -08002777 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002778 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2779
Colin Crossae8600b2020-10-29 17:09:13 -07002780 ctx := testContext(config)
Colin Cross50ddcc42019-05-16 12:28:22 -07002781
2782 run(t, ctx, config)
2783
2784 app := ctx.ModuleForTests("app", "android_common")
2785 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2786
Paul Duffin859fe962020-05-15 10:20:31 +01002787 // Test that implicit dependencies on java_sdk_library instances are passed to the manifest.
2788 manifestFixerArgs := app.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
Ulya Trafimovich4b6d4c12020-08-19 14:58:01 +01002789 for _, w := range []string{"qux", "quuz", "runtime-library"} {
2790 if !strings.Contains(manifestFixerArgs, "--uses-library "+w) {
2791 t.Errorf("unexpected manifest_fixer args: wanted %q in %q", w, manifestFixerArgs)
2792 }
Paul Duffin859fe962020-05-15 10:20:31 +01002793 }
2794
Colin Cross50ddcc42019-05-16 12:28:22 -07002795 // Test that all libraries are verified
2796 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2797 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2798 t.Errorf("wanted %q in %q", w, cmd)
2799 }
2800
2801 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2802 t.Errorf("wanted %q in %q", w, cmd)
2803 }
2804
2805 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2806
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002807 if w := `uses_library_names="foo android.test.runner"`; !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002808 t.Errorf("wanted %q in %q", w, cmd)
2809 }
2810
2811 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2812 t.Errorf("wanted %q in %q", w, cmd)
2813 }
2814
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002815 // Test that all present libraries are preopted, including implicit SDK dependencies, possibly stubs
Colin Cross50ddcc42019-05-16 12:28:22 -07002816 cmd = app.Rule("dexpreopt").RuleParams.Command
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002817 w := `--target-context-for-sdk any ` +
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002818 `PCL[/system/framework/qux.jar]#` +
Ulya Trafimovich8cbc5d22020-11-03 15:15:46 +00002819 `PCL[/system/framework/quuz.jar]#` +
2820 `PCL[/system/framework/foo.jar]#` +
2821 `PCL[/system/framework/bar.jar]#` +
2822 `PCL[/system/framework/runtime-library.jar]`
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002823 if !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002824 t.Errorf("wanted %q in %q", w, cmd)
2825 }
2826
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002827 // Test conditional context for target SDK version 28.
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002828 if w := `--target-context-for-sdk 28` +
2829 ` PCL[/system/framework/org.apache.http.legacy.jar] `; !strings.Contains(cmd, w) {
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002830 t.Errorf("wanted %q in %q", w, cmd)
2831 }
2832
2833 // Test conditional context for target SDK version 29.
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002834 if w := `--target-context-for-sdk 29` +
2835 ` PCL[/system/framework/android.hidl.base-V1.0-java.jar]` +
Ulya Trafimovich180fece2020-10-28 10:58:25 +00002836 `#PCL[/system/framework/android.hidl.manager-V1.0-java.jar] `; !strings.Contains(cmd, w) {
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002837 t.Errorf("wanted %q in %q", w, cmd)
2838 }
2839
2840 // Test conditional context for target SDK version 30.
Ulya Trafimovich46b3d5b2020-10-21 13:20:55 +01002841 // "android.test.mock" is absent because "android.test.runner" is not used.
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002842 if w := `--target-context-for-sdk 30` +
2843 ` PCL[/system/framework/android.test.base.jar] `; !strings.Contains(cmd, w) {
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002844 t.Errorf("wanted %q in %q", w, cmd)
2845 }
2846
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +01002847 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002848 if w := `--target-context-for-sdk any` +
2849 ` PCL[/system/framework/foo.jar]` +
2850 `#PCL[/system/framework/android.test.runner.jar]` +
2851 `#PCL[/system/framework/bar.jar] `; !strings.Contains(cmd, w) {
Ulya Trafimovich24813e12020-10-07 15:05:21 +01002852 t.Errorf("wanted %q in %q", w, cmd)
2853 }
2854
2855 // Test conditional context for target SDK version 30.
Ulya Trafimovich46b3d5b2020-10-21 13:20:55 +01002856 // "android.test.mock" is present because "android.test.runner" is used.
Ulya Trafimovich8130c482020-10-07 15:17:13 +01002857 if w := `--target-context-for-sdk 30` +
2858 ` PCL[/system/framework/android.test.base.jar]` +
2859 `#PCL[/system/framework/android.test.mock.jar] `; !strings.Contains(cmd, w) {
Colin Cross50ddcc42019-05-16 12:28:22 -07002860 t.Errorf("wanted %q in %q", w, cmd)
2861 }
2862}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002863
2864func TestCodelessApp(t *testing.T) {
2865 testCases := []struct {
2866 name string
2867 bp string
2868 noCode bool
2869 }{
2870 {
2871 name: "normal",
2872 bp: `
2873 android_app {
2874 name: "foo",
2875 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002876 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002877 }
2878 `,
2879 noCode: false,
2880 },
2881 {
2882 name: "app without sources",
2883 bp: `
2884 android_app {
2885 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002886 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002887 }
2888 `,
2889 noCode: true,
2890 },
2891 {
2892 name: "app with libraries",
2893 bp: `
2894 android_app {
2895 name: "foo",
2896 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002897 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002898 }
2899
2900 java_library {
2901 name: "lib",
2902 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002903 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002904 }
2905 `,
2906 noCode: false,
2907 },
2908 {
2909 name: "app with sourceless libraries",
2910 bp: `
2911 android_app {
2912 name: "foo",
2913 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002914 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002915 }
2916
2917 java_library {
2918 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002919 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002920 }
2921 `,
2922 // TODO(jungjw): this should probably be true
2923 noCode: false,
2924 },
2925 }
2926
2927 for _, test := range testCases {
2928 t.Run(test.name, func(t *testing.T) {
2929 ctx := testApp(t, test.bp)
2930
2931 foo := ctx.ModuleForTests("foo", "android_common")
2932 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2933 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2934 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2935 }
2936 })
2937 }
2938}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002939
2940func TestEmbedNotice(t *testing.T) {
Colin Cross238c1f32020-06-07 16:58:18 -07002941 ctx, _ := testJavaWithFS(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002942 android_app {
2943 name: "foo",
2944 srcs: ["a.java"],
2945 static_libs: ["javalib"],
2946 jni_libs: ["libjni"],
2947 notice: "APP_NOTICE",
2948 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002949 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002950 }
2951
2952 // No embed_notice flag
2953 android_app {
2954 name: "bar",
2955 srcs: ["a.java"],
2956 jni_libs: ["libjni"],
2957 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002958 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002959 }
2960
2961 // No NOTICE files
2962 android_app {
2963 name: "baz",
2964 srcs: ["a.java"],
2965 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002966 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002967 }
2968
2969 cc_library {
2970 name: "libjni",
2971 system_shared_libs: [],
2972 stl: "none",
2973 notice: "LIB_NOTICE",
Colin Cross094cde42020-02-15 10:38:00 -08002974 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002975 }
2976
2977 java_library {
2978 name: "javalib",
2979 srcs: [
2980 ":gen",
2981 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002982 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002983 }
2984
2985 genrule {
2986 name: "gen",
2987 tools: ["gentool"],
2988 out: ["gen.java"],
2989 notice: "GENRULE_NOTICE",
2990 }
2991
2992 java_binary_host {
2993 name: "gentool",
2994 srcs: ["b.java"],
2995 notice: "TOOL_NOTICE",
2996 }
Colin Cross238c1f32020-06-07 16:58:18 -07002997 `, map[string][]byte{
2998 "APP_NOTICE": nil,
2999 "GENRULE_NOTICE": nil,
3000 "LIB_NOTICE": nil,
3001 "TOOL_NOTICE": nil,
3002 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -07003003
3004 // foo has NOTICE files to process, and embed_notices is true.
3005 foo := ctx.ModuleForTests("foo", "android_common")
3006 // verify merge notices rule.
3007 mergeNotices := foo.Rule("mergeNoticesRule")
3008 noticeInputs := mergeNotices.Inputs.Strings()
3009 // TOOL_NOTICE should be excluded as it's a host module.
3010 if len(mergeNotices.Inputs) != 3 {
3011 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
3012 }
3013 if !inList("APP_NOTICE", noticeInputs) {
3014 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
3015 }
3016 if !inList("LIB_NOTICE", noticeInputs) {
3017 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
3018 }
3019 if !inList("GENRULE_NOTICE", noticeInputs) {
3020 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
3021 }
3022 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
3023 res := foo.Output("package-res.apk")
3024 aapt2Flags := res.Args["flags"]
3025 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
3026 if !strings.Contains(aapt2Flags, e) {
3027 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
3028 }
3029
3030 // bar has NOTICE files to process, but embed_notices is not set.
3031 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07003032 res = bar.Output("package-res.apk")
3033 aapt2Flags = res.Args["flags"]
3034 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
3035 if strings.Contains(aapt2Flags, e) {
3036 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07003037 }
3038
3039 // baz's embed_notice is true, but it doesn't have any NOTICE files.
3040 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07003041 res = baz.Output("package-res.apk")
3042 aapt2Flags = res.Args["flags"]
3043 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
3044 if strings.Contains(aapt2Flags, e) {
3045 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07003046 }
3047}
Colin Cross53a87f52019-06-25 13:35:30 -07003048
3049func TestUncompressDex(t *testing.T) {
3050 testCases := []struct {
3051 name string
3052 bp string
3053
3054 uncompressedPlatform bool
3055 uncompressedUnbundled bool
3056 }{
3057 {
3058 name: "normal",
3059 bp: `
3060 android_app {
3061 name: "foo",
3062 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09003063 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07003064 }
3065 `,
3066 uncompressedPlatform: true,
3067 uncompressedUnbundled: false,
3068 },
3069 {
3070 name: "use_embedded_dex",
3071 bp: `
3072 android_app {
3073 name: "foo",
3074 use_embedded_dex: true,
3075 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09003076 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07003077 }
3078 `,
3079 uncompressedPlatform: true,
3080 uncompressedUnbundled: true,
3081 },
3082 {
3083 name: "privileged",
3084 bp: `
3085 android_app {
3086 name: "foo",
3087 privileged: true,
3088 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09003089 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07003090 }
3091 `,
3092 uncompressedPlatform: true,
3093 uncompressedUnbundled: true,
3094 },
David Srbeckye033cba2020-05-20 22:20:28 +01003095 {
3096 name: "normal_uncompress_dex_true",
3097 bp: `
3098 android_app {
3099 name: "foo",
3100 srcs: ["a.java"],
3101 sdk_version: "current",
3102 uncompress_dex: true,
3103 }
3104 `,
3105 uncompressedPlatform: true,
3106 uncompressedUnbundled: true,
3107 },
3108 {
3109 name: "normal_uncompress_dex_false",
3110 bp: `
3111 android_app {
3112 name: "foo",
3113 srcs: ["a.java"],
3114 sdk_version: "current",
3115 uncompress_dex: false,
3116 }
3117 `,
3118 uncompressedPlatform: false,
3119 uncompressedUnbundled: false,
3120 },
Colin Cross53a87f52019-06-25 13:35:30 -07003121 }
3122
3123 test := func(t *testing.T, bp string, want bool, unbundled bool) {
3124 t.Helper()
3125
Colin Cross98be1bb2019-12-13 20:41:13 -08003126 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07003127 if unbundled {
3128 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
Jeongik Cha816a23a2020-07-08 01:09:23 +09003129 config.TestProductVariables.Always_use_prebuilt_sdks = proptools.BoolPtr(true)
Colin Cross53a87f52019-06-25 13:35:30 -07003130 }
3131
Colin Crossae8600b2020-10-29 17:09:13 -07003132 ctx := testContext(config)
Colin Cross53a87f52019-06-25 13:35:30 -07003133
3134 run(t, ctx, config)
3135
3136 foo := ctx.ModuleForTests("foo", "android_common")
3137 dex := foo.Rule("r8")
3138 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
3139 aligned := foo.MaybeRule("zipalign").Rule != nil
3140
3141 if uncompressedInDexJar != want {
3142 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
3143 }
3144
3145 if aligned != want {
3146 t.Errorf("want aligned %v, got %v", want, aligned)
3147 }
3148 }
3149
3150 for _, tt := range testCases {
3151 t.Run(tt.name, func(t *testing.T) {
3152 t.Run("platform", func(t *testing.T) {
3153 test(t, tt.bp, tt.uncompressedPlatform, false)
3154 })
3155 t.Run("unbundled", func(t *testing.T) {
3156 test(t, tt.bp, tt.uncompressedUnbundled, true)
3157 })
3158 })
3159 }
3160}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07003161
3162func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
3163 if expectedValue != "" {
3164 expectedFlag := "--" + flagName + " " + expectedValue
3165 if !strings.Contains(aapt2Flags, expectedFlag) {
3166 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
3167 }
3168 } else {
3169 unexpectedFlag := "--" + flagName
3170 if strings.Contains(aapt2Flags, unexpectedFlag) {
3171 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
3172 }
3173 }
3174}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003175
3176func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003177 fs := map[string][]byte{
3178 "baz/res/res/values/strings.xml": nil,
3179 "bar/res/res/values/strings.xml": nil,
3180 }
3181 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003182 runtime_resource_overlay {
3183 name: "foo",
3184 certificate: "platform",
Liz Kammer966b2f02020-05-19 16:15:25 -07003185 lineage: "lineage.bin",
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003186 product_specific: true,
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003187 static_libs: ["bar"],
3188 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003189 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003190 }
3191
3192 runtime_resource_overlay {
3193 name: "foo_themed",
3194 certificate: "platform",
3195 product_specific: true,
3196 theme: "faza",
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003197 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003198 }
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003199
3200 android_library {
3201 name: "bar",
3202 resource_dirs: ["bar/res"],
3203 }
3204
3205 android_app {
3206 name: "baz",
3207 sdk_version: "current",
3208 resource_dirs: ["baz/res"],
3209 }
3210 `
3211 config := testAppConfig(nil, bp, fs)
Colin Crossae8600b2020-10-29 17:09:13 -07003212 ctx := testContext(config)
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003213 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003214
3215 m := ctx.ModuleForTests("foo", "android_common")
3216
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08003217 // Check AAPT2 link flags.
3218 aapt2Flags := m.Output("package-res.apk").Args["flags"]
3219 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3220 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
3221 if len(absentFlags) > 0 {
3222 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3223 }
3224
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07003225 // Check overlay.list output for static_libs dependency.
3226 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
3227 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
3228 if !inList(staticLibPackage, overlayList) {
3229 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
3230 }
3231
3232 // Check AAPT2 link flags for resource_libs dependency.
3233 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
3234 if !strings.Contains(aapt2Flags, resourceLibFlag) {
3235 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
3236 }
3237
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003238 // Check cert signing flag.
3239 signedApk := m.Output("signed/foo.apk")
Liz Kammer966b2f02020-05-19 16:15:25 -07003240 lineageFlag := signedApk.Args["flags"]
3241 expectedLineageFlag := "--lineage lineage.bin"
3242 if expectedLineageFlag != lineageFlag {
3243 t.Errorf("Incorrect signing lineage flags, expected: %q, got: %q", expectedLineageFlag, lineageFlag)
3244 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003245 signingFlag := signedApk.Args["certificates"]
3246 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
3247 if expected != signingFlag {
3248 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
3249 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003250 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3251 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003252 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
3253 if !reflect.DeepEqual(path, expectedPath) {
3254 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
3255 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003256
3257 // Check device location.
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003258 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08003259 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003260 if !reflect.DeepEqual(path, expectedPath) {
3261 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3262 }
3263
3264 // A themed module has a different device location
3265 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003266 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
3267 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003268 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
3269 if !reflect.DeepEqual(path, expectedPath) {
3270 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3271 }
Jaewoong Jungad0177b2020-04-24 15:22:40 -07003272
3273 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
3274 expectedOverrides := []string{"foo"}
3275 if !reflect.DeepEqual(overrides, expectedOverrides) {
3276 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
3277 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08003278}
Roshan Pius4df2bc72020-04-27 09:42:27 -07003279
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07003280func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
3281 ctx, config := testJava(t, `
3282 java_defaults {
3283 name: "rro_defaults",
3284 theme: "default_theme",
3285 product_specific: true,
3286 aaptflags: ["--keep-raw-values"],
3287 }
3288
3289 runtime_resource_overlay {
3290 name: "foo_with_defaults",
3291 defaults: ["rro_defaults"],
3292 }
3293
3294 runtime_resource_overlay {
3295 name: "foo_barebones",
3296 }
3297 `)
3298
3299 //
3300 // RRO module with defaults
3301 //
3302 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
3303
3304 // Check AAPT2 link flags.
3305 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3306 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
3307 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
3308 if len(absentFlags) > 0 {
3309 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
3310 }
3311
3312 // Check device location.
3313 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3314 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
3315 if !reflect.DeepEqual(path, expectedPath) {
3316 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
3317 }
3318
3319 //
3320 // RRO module without defaults
3321 //
3322 m = ctx.ModuleForTests("foo_barebones", "android_common")
3323
3324 // Check AAPT2 link flags.
3325 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
3326 unexpectedFlags := "--keep-raw-values"
3327 if inList(unexpectedFlags, aapt2Flags) {
3328 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
3329 }
3330
3331 // Check device location.
3332 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
3333 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
3334 if !reflect.DeepEqual(path, expectedPath) {
3335 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
3336 }
3337}
Roshan Piusb8307962020-04-27 09:42:27 -07003338
Roshan Pius4df2bc72020-04-27 09:42:27 -07003339func TestOverrideRuntimeResourceOverlay(t *testing.T) {
3340 ctx, _ := testJava(t, `
3341 runtime_resource_overlay {
3342 name: "foo_overlay",
3343 certificate: "platform",
3344 product_specific: true,
3345 sdk_version: "current",
3346 }
3347
3348 override_runtime_resource_overlay {
3349 name: "bar_overlay",
3350 base: "foo_overlay",
3351 package_name: "com.android.bar.overlay",
3352 target_package_name: "com.android.bar",
3353 }
3354 `)
3355
3356 expectedVariants := []struct {
3357 moduleName string
3358 variantName string
3359 apkPath string
3360 overrides []string
3361 targetVariant string
3362 packageFlag string
3363 targetPackageFlag string
3364 }{
3365 {
3366 variantName: "android_common",
3367 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
3368 overrides: nil,
3369 targetVariant: "android_common",
3370 packageFlag: "",
3371 targetPackageFlag: "",
3372 },
3373 {
3374 variantName: "android_common_bar_overlay",
3375 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
3376 overrides: []string{"foo_overlay"},
3377 targetVariant: "android_common_bar",
3378 packageFlag: "com.android.bar.overlay",
3379 targetPackageFlag: "com.android.bar",
3380 },
3381 }
3382 for _, expected := range expectedVariants {
3383 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
3384
3385 // Check the final apk name
3386 outputs := variant.AllOutputs()
3387 expectedApkPath := buildDir + expected.apkPath
3388 found := false
3389 for _, o := range outputs {
3390 if o == expectedApkPath {
3391 found = true
3392 break
3393 }
3394 }
3395 if !found {
3396 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
3397 }
3398
3399 // Check if the overrides field values are correctly aggregated.
3400 mod := variant.Module().(*RuntimeResourceOverlay)
3401 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
3402 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
3403 expected.overrides, mod.properties.Overrides)
3404 }
3405
3406 // Check aapt2 flags.
3407 res := variant.Output("package-res.apk")
3408 aapt2Flags := res.Args["flags"]
3409 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
Liz Kammer9f9fd022020-06-18 19:44:06 +00003410 checkAapt2LinkFlag(t, aapt2Flags, "rename-resources-package", "")
Roshan Pius4df2bc72020-04-27 09:42:27 -07003411 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
3412 }
3413}
Jaewoong Jungc779cd42020-10-06 18:56:10 -07003414
3415func TestEnforceRRO_propagatesToDependencies(t *testing.T) {
3416 testCases := []struct {
3417 name string
3418 enforceRROTargets []string
3419 enforceRROExemptTargets []string
3420 rroDirs map[string][]string
3421 }{
3422 {
3423 name: "no RRO",
3424 enforceRROTargets: nil,
3425 enforceRROExemptTargets: nil,
3426 rroDirs: map[string][]string{
3427 "foo": nil,
3428 "bar": nil,
3429 },
3430 },
3431 {
3432 name: "enforce RRO on all",
3433 enforceRROTargets: []string{"*"},
3434 enforceRROExemptTargets: nil,
3435 rroDirs: map[string][]string{
3436 "foo": {"product/vendor/blah/overlay/lib2/res"},
3437 "bar": {"product/vendor/blah/overlay/lib2/res"},
3438 },
3439 },
3440 {
3441 name: "enforce RRO on foo",
3442 enforceRROTargets: []string{"foo"},
3443 enforceRROExemptTargets: nil,
3444 rroDirs: map[string][]string{
3445 "foo": {"product/vendor/blah/overlay/lib2/res"},
3446 "bar": {"product/vendor/blah/overlay/lib2/res"},
3447 },
3448 },
3449 {
3450 name: "enforce RRO on foo, bar exempted",
3451 enforceRROTargets: []string{"foo"},
3452 enforceRROExemptTargets: []string{"bar"},
3453 rroDirs: map[string][]string{
3454 "foo": {"product/vendor/blah/overlay/lib2/res"},
3455 "bar": nil,
3456 },
3457 },
3458 }
3459
3460 productResourceOverlays := []string{
3461 "product/vendor/blah/overlay",
3462 }
3463
3464 fs := map[string][]byte{
3465 "lib2/res/values/strings.xml": nil,
3466 "product/vendor/blah/overlay/lib2/res/values/strings.xml": nil,
3467 }
3468
3469 bp := `
3470 android_app {
3471 name: "foo",
3472 sdk_version: "current",
3473 resource_dirs: [],
3474 static_libs: ["lib"],
3475 }
3476
3477 android_app {
3478 name: "bar",
3479 sdk_version: "current",
3480 resource_dirs: [],
3481 static_libs: ["lib"],
3482 }
3483
3484 android_library {
3485 name: "lib",
3486 sdk_version: "current",
3487 resource_dirs: [],
3488 static_libs: ["lib2"],
3489 }
3490
3491 android_library {
3492 name: "lib2",
3493 sdk_version: "current",
3494 resource_dirs: ["lib2/res"],
3495 }
3496 `
3497
3498 for _, testCase := range testCases {
3499 t.Run(testCase.name, func(t *testing.T) {
3500 config := testAppConfig(nil, bp, fs)
3501 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
3502 if testCase.enforceRROTargets != nil {
3503 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
3504 }
3505 if testCase.enforceRROExemptTargets != nil {
3506 config.TestProductVariables.EnforceRROExemptedTargets = testCase.enforceRROExemptTargets
3507 }
3508
Colin Crossae8600b2020-10-29 17:09:13 -07003509 ctx := testContext(config)
Jaewoong Jungc779cd42020-10-06 18:56:10 -07003510 run(t, ctx, config)
3511
3512 modules := []string{"foo", "bar"}
3513 for _, moduleName := range modules {
3514 module := ctx.ModuleForTests(moduleName, "android_common")
3515 mkEntries := android.AndroidMkEntriesForTest(t, config, "", module.Module())[0]
3516 actualRRODirs := mkEntries.EntryMap["LOCAL_SOONG_PRODUCT_RRO_DIRS"]
3517 if !reflect.DeepEqual(actualRRODirs, testCase.rroDirs[moduleName]) {
3518 t.Errorf("exected %s LOCAL_SOONG_PRODUCT_RRO_DIRS entry: %v\ngot:%q",
3519 moduleName, testCase.rroDirs[moduleName], actualRRODirs)
3520 }
3521 }
3522 })
3523 }
3524}
Cole Faust9a631312020-10-22 21:05:24 +00003525
3526func TestExportedProguardFlagFiles(t *testing.T) {
3527 ctx, _ := testJava(t, `
3528 android_app {
3529 name: "foo",
3530 sdk_version: "current",
3531 static_libs: ["lib1"],
3532 }
3533
3534 android_library {
3535 name: "lib1",
3536 sdk_version: "current",
3537 optimize: {
3538 proguard_flags_files: ["lib1proguard.cfg"],
3539 }
3540 }
3541 `)
3542
3543 m := ctx.ModuleForTests("foo", "android_common")
3544 hasLib1Proguard := false
3545 for _, s := range m.Rule("java.r8").Implicits.Strings() {
3546 if s == "lib1proguard.cfg" {
3547 hasLib1Proguard = true
3548 break
3549 }
3550 }
3551
3552 if !hasLib1Proguard {
3553 t.Errorf("App does not use library proguard config")
3554 }
3555}