blob: 7d47a5c2fe809858e29e6214b22fc27883cf1d51 [file] [log] [blame]
Colin Cross3bc7ffa2017-11-22 16:19:37 -08001// Copyright 2017 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package java
16
17import (
Colin Crossd09b0b62018-04-18 11:06:47 -070018 "fmt"
Colin Crossa4f08812018-10-02 22:03:40 -070019 "path/filepath"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080020 "reflect"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070021 "regexp"
Colin Crossb69301e2017-12-01 10:48:26 -080022 "sort"
Colin Crossd09b0b62018-04-18 11:06:47 -070023 "strings"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080024 "testing"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070025
26 "github.com/google/blueprint/proptools"
27
28 "android/soong/android"
29 "android/soong/cc"
Colin Cross3bc7ffa2017-11-22 16:19:37 -080030)
31
32var (
33 resourceFiles = []string{
34 "res/layout/layout.xml",
35 "res/values/strings.xml",
36 "res/values-en-rUS/strings.xml",
37 }
38
39 compiledResourceFiles = []string{
40 "aapt2/res/layout_layout.xml.flat",
41 "aapt2/res/values_strings.arsc.flat",
42 "aapt2/res/values-en-rUS_strings.arsc.flat",
43 }
44)
45
Colin Cross98be1bb2019-12-13 20:41:13 -080046func testAppConfig(env map[string]string, bp string, fs map[string][]byte) android.Config {
Colin Cross527012a2017-11-30 22:56:16 -080047 appFS := map[string][]byte{}
48 for k, v := range fs {
49 appFS[k] = v
Colin Cross3bc7ffa2017-11-22 16:19:37 -080050 }
51
Colin Cross527012a2017-11-30 22:56:16 -080052 for _, file := range resourceFiles {
53 appFS[file] = nil
54 }
55
Colin Cross98be1bb2019-12-13 20:41:13 -080056 return testConfig(env, bp, appFS)
Colin Cross527012a2017-11-30 22:56:16 -080057}
58
59func testApp(t *testing.T, bp string) *android.TestContext {
Colin Cross98be1bb2019-12-13 20:41:13 -080060 config := testAppConfig(nil, bp, nil)
Colin Cross527012a2017-11-30 22:56:16 -080061
Colin Cross98be1bb2019-12-13 20:41:13 -080062 ctx := testContext()
Colin Cross527012a2017-11-30 22:56:16 -080063
64 run(t, ctx, config)
65
66 return ctx
Colin Cross3bc7ffa2017-11-22 16:19:37 -080067}
68
69func TestApp(t *testing.T) {
Colin Crossa97c5d32018-03-28 14:58:31 -070070 for _, moduleType := range []string{"android_app", "android_library"} {
71 t.Run(moduleType, func(t *testing.T) {
72 ctx := testApp(t, moduleType+` {
73 name: "foo",
74 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +090075 sdk_version: "current"
Colin Crossa97c5d32018-03-28 14:58:31 -070076 }
77 `)
Colin Cross3bc7ffa2017-11-22 16:19:37 -080078
Colin Crossa97c5d32018-03-28 14:58:31 -070079 foo := ctx.ModuleForTests("foo", "android_common")
Colin Cross3bc7ffa2017-11-22 16:19:37 -080080
Colin Cross31656952018-05-24 16:11:20 -070081 var expectedLinkImplicits []string
82
83 manifestFixer := foo.Output("manifest_fixer/AndroidManifest.xml")
84 expectedLinkImplicits = append(expectedLinkImplicits, manifestFixer.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080085
Colin Crossa97c5d32018-03-28 14:58:31 -070086 frameworkRes := ctx.ModuleForTests("framework-res", "android_common")
87 expectedLinkImplicits = append(expectedLinkImplicits,
88 frameworkRes.Output("package-res.apk").Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -080089
Colin Crossa97c5d32018-03-28 14:58:31 -070090 // Test the mapping from input files to compiled output file names
91 compile := foo.Output(compiledResourceFiles[0])
92 if !reflect.DeepEqual(resourceFiles, compile.Inputs.Strings()) {
93 t.Errorf("expected aapt2 compile inputs expected:\n %#v\n got:\n %#v",
94 resourceFiles, compile.Inputs.Strings())
95 }
Colin Crossb69301e2017-12-01 10:48:26 -080096
Colin Crossa97c5d32018-03-28 14:58:31 -070097 compiledResourceOutputs := compile.Outputs.Strings()
98 sort.Strings(compiledResourceOutputs)
Colin Crossb69301e2017-12-01 10:48:26 -080099
Colin Crossa97c5d32018-03-28 14:58:31 -0700100 expectedLinkImplicits = append(expectedLinkImplicits, compiledResourceOutputs...)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800101
Colin Crossa97c5d32018-03-28 14:58:31 -0700102 list := foo.Output("aapt2/res.list")
103 expectedLinkImplicits = append(expectedLinkImplicits, list.Output.String())
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800104
Colin Crossa97c5d32018-03-28 14:58:31 -0700105 // Check that the link rule uses
106 res := ctx.ModuleForTests("foo", "android_common").Output("package-res.apk")
107 if !reflect.DeepEqual(expectedLinkImplicits, res.Implicits.Strings()) {
108 t.Errorf("expected aapt2 link implicits expected:\n %#v\n got:\n %#v",
109 expectedLinkImplicits, res.Implicits.Strings())
110 }
111 })
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800112 }
113}
Colin Cross890ff552017-11-30 20:13:19 -0800114
Colin Crosse560c4a2019-03-19 16:03:11 -0700115func TestAppSplits(t *testing.T) {
116 ctx := testApp(t, `
117 android_app {
118 name: "foo",
119 srcs: ["a.java"],
120 package_splits: ["v4", "v7,hdpi"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900121 sdk_version: "current"
Colin Crosse560c4a2019-03-19 16:03:11 -0700122 }`)
123
124 foo := ctx.ModuleForTests("foo", "android_common")
125
126 expectedOutputs := []string{
127 filepath.Join(buildDir, ".intermediates/foo/android_common/foo.apk"),
128 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v4.apk"),
129 filepath.Join(buildDir, ".intermediates/foo/android_common/foo_v7_hdpi.apk"),
130 }
131 for _, expectedOutput := range expectedOutputs {
132 foo.Output(expectedOutput)
133 }
134
Colin Cross41955e82019-05-29 14:40:35 -0700135 outputFiles, err := foo.Module().(*AndroidApp).OutputFiles("")
136 if err != nil {
137 t.Fatal(err)
138 }
139 if g, w := outputFiles.Strings(), expectedOutputs; !reflect.DeepEqual(g, w) {
140 t.Errorf(`want OutputFiles("") = %q, got %q`, w, g)
Colin Crosse560c4a2019-03-19 16:03:11 -0700141 }
142}
143
Jeongik Cha538c0d02019-07-11 15:54:27 +0900144func TestPlatformAPIs(t *testing.T) {
145 testJava(t, `
146 android_app {
147 name: "foo",
148 srcs: ["a.java"],
149 platform_apis: true,
150 }
151 `)
152
153 testJava(t, `
154 android_app {
155 name: "foo",
156 srcs: ["a.java"],
157 sdk_version: "current",
158 }
159 `)
160
161 testJavaError(t, "platform_apis must be true when sdk_version is empty.", `
162 android_app {
163 name: "bar",
164 srcs: ["b.java"],
165 }
166 `)
167
168 testJavaError(t, "platform_apis must be false when sdk_version is not empty.", `
169 android_app {
170 name: "bar",
171 srcs: ["b.java"],
172 sdk_version: "system_current",
173 platform_apis: true,
174 }
175 `)
176}
177
Jeongik Chae403e9e2019-12-07 00:16:24 +0900178func TestAndroidAppLinkType(t *testing.T) {
179 testJava(t, `
180 android_app {
181 name: "foo",
182 srcs: ["a.java"],
183 libs: ["bar"],
184 static_libs: ["baz"],
185 platform_apis: true,
186 }
187
188 java_library {
189 name: "bar",
190 sdk_version: "current",
191 srcs: ["b.java"],
192 }
193
194 android_library {
195 name: "baz",
196 sdk_version: "system_current",
197 srcs: ["c.java"],
198 }
199 `)
200
201 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
202 android_app {
203 name: "foo",
204 srcs: ["a.java"],
205 libs: ["bar"],
206 sdk_version: "current",
207 static_libs: ["baz"],
208 }
209
210 java_library {
211 name: "bar",
212 sdk_version: "current",
213 srcs: ["b.java"],
214 }
215
216 android_library {
217 name: "baz",
218 sdk_version: "system_current",
219 srcs: ["c.java"],
220 }
221 `)
222
223 testJava(t, `
224 android_app {
225 name: "foo",
226 srcs: ["a.java"],
227 libs: ["bar"],
228 sdk_version: "system_current",
229 static_libs: ["baz"],
230 }
231
232 java_library {
233 name: "bar",
234 sdk_version: "current",
235 srcs: ["b.java"],
236 }
237
238 android_library {
239 name: "baz",
240 sdk_version: "system_current",
241 srcs: ["c.java"],
242 }
243 `)
244
245 testJavaError(t, "Adjust sdk_version: property of the source or target module so that target module is built with the same or smaller API set than the source.", `
246 android_app {
247 name: "foo",
248 srcs: ["a.java"],
249 libs: ["bar"],
250 sdk_version: "system_current",
251 static_libs: ["baz"],
252 }
253
254 java_library {
255 name: "bar",
256 sdk_version: "current",
257 srcs: ["b.java"],
258 }
259
260 android_library {
261 name: "baz",
262 srcs: ["c.java"],
263 }
264 `)
265}
266
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100267func TestUpdatableApps(t *testing.T) {
268 testCases := []struct {
269 name string
270 bp string
271 expectedError string
272 }{
273 {
274 name: "Stable public SDK",
275 bp: `android_app {
276 name: "foo",
277 srcs: ["a.java"],
278 sdk_version: "29",
Artur Satayev11962102020-04-16 13:43:02 +0100279 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100280 updatable: true,
281 }`,
282 },
283 {
284 name: "Stable system SDK",
285 bp: `android_app {
286 name: "foo",
287 srcs: ["a.java"],
288 sdk_version: "system_29",
Artur Satayev11962102020-04-16 13:43:02 +0100289 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100290 updatable: true,
291 }`,
292 },
293 {
294 name: "Current public SDK",
295 bp: `android_app {
296 name: "foo",
297 srcs: ["a.java"],
298 sdk_version: "current",
Artur Satayev11962102020-04-16 13:43:02 +0100299 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100300 updatable: true,
301 }`,
302 },
303 {
304 name: "Current system SDK",
305 bp: `android_app {
306 name: "foo",
307 srcs: ["a.java"],
308 sdk_version: "system_current",
Artur Satayev11962102020-04-16 13:43:02 +0100309 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100310 updatable: true,
311 }`,
312 },
313 {
314 name: "Current module SDK",
315 bp: `android_app {
316 name: "foo",
317 srcs: ["a.java"],
318 sdk_version: "module_current",
Artur Satayev11962102020-04-16 13:43:02 +0100319 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100320 updatable: true,
321 }`,
322 },
323 {
324 name: "Current core SDK",
325 bp: `android_app {
326 name: "foo",
327 srcs: ["a.java"],
328 sdk_version: "core_current",
Artur Satayev11962102020-04-16 13:43:02 +0100329 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100330 updatable: true,
331 }`,
332 },
333 {
334 name: "No Platform APIs",
335 bp: `android_app {
336 name: "foo",
337 srcs: ["a.java"],
338 platform_apis: true,
Artur Satayev11962102020-04-16 13:43:02 +0100339 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100340 updatable: true,
341 }`,
342 expectedError: "Updatable apps must use stable SDKs",
343 },
344 {
345 name: "No Core Platform APIs",
346 bp: `android_app {
347 name: "foo",
348 srcs: ["a.java"],
349 sdk_version: "core_platform",
Artur Satayev11962102020-04-16 13:43:02 +0100350 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100351 updatable: true,
352 }`,
353 expectedError: "Updatable apps must use stable SDKs",
354 },
355 {
356 name: "No unspecified APIs",
357 bp: `android_app {
358 name: "foo",
359 srcs: ["a.java"],
360 updatable: true,
Artur Satayev11962102020-04-16 13:43:02 +0100361 min_sdk_version: "29",
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100362 }`,
363 expectedError: "Updatable apps must use stable SDK",
364 },
Artur Satayev11962102020-04-16 13:43:02 +0100365 {
366 name: "Must specify min_sdk_version",
367 bp: `android_app {
368 name: "app_without_min_sdk_version",
369 srcs: ["a.java"],
370 sdk_version: "29",
371 updatable: true,
372 }`,
373 expectedError: "updatable apps must set min_sdk_version.",
374 },
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100375 }
376
377 for _, test := range testCases {
378 t.Run(test.name, func(t *testing.T) {
379 if test.expectedError == "" {
380 testJava(t, test.bp)
381 } else {
382 testJavaError(t, test.expectedError, test.bp)
383 }
384 })
385 }
386}
387
Colin Cross0ddae7f2019-02-07 15:30:01 -0800388func TestResourceDirs(t *testing.T) {
389 testCases := []struct {
390 name string
391 prop string
392 resources []string
393 }{
394 {
395 name: "no resource_dirs",
396 prop: "",
397 resources: []string{"res/res/values/strings.xml"},
398 },
399 {
400 name: "resource_dirs",
401 prop: `resource_dirs: ["res"]`,
402 resources: []string{"res/res/values/strings.xml"},
403 },
404 {
405 name: "empty resource_dirs",
406 prop: `resource_dirs: []`,
407 resources: nil,
408 },
409 }
410
411 fs := map[string][]byte{
412 "res/res/values/strings.xml": nil,
413 }
414
415 bp := `
416 android_app {
417 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900418 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800419 %s
420 }
421 `
422
423 for _, testCase := range testCases {
424 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800425 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
426 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800427 run(t, ctx, config)
428
429 module := ctx.ModuleForTests("foo", "android_common")
430 resourceList := module.MaybeOutput("aapt2/res.list")
431
432 var resources []string
433 if resourceList.Rule != nil {
434 for _, compiledResource := range resourceList.Inputs.Strings() {
435 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
436 }
437 }
438
439 if !reflect.DeepEqual(resources, testCase.resources) {
440 t.Errorf("expected resource files %q, got %q",
441 testCase.resources, resources)
442 }
443 })
444 }
445}
446
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800447func TestLibraryAssets(t *testing.T) {
448 bp := `
449 android_app {
450 name: "foo",
451 sdk_version: "current",
452 static_libs: ["lib1", "lib2", "lib3"],
453 }
454
455 android_library {
456 name: "lib1",
457 sdk_version: "current",
458 asset_dirs: ["assets_a"],
459 }
460
461 android_library {
462 name: "lib2",
463 sdk_version: "current",
464 }
465
466 android_library {
467 name: "lib3",
468 sdk_version: "current",
469 static_libs: ["lib4"],
470 }
471
472 android_library {
473 name: "lib4",
474 sdk_version: "current",
475 asset_dirs: ["assets_b"],
476 }
477 `
478
479 testCases := []struct {
480 name string
481 assetFlag string
482 assetPackages []string
483 }{
484 {
485 name: "foo",
486 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
487 assetPackages: []string{
488 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
489 buildDir + "/.intermediates/lib1/android_common/assets.zip",
490 buildDir + "/.intermediates/lib3/android_common/assets.zip",
491 },
492 },
493 {
494 name: "lib1",
495 assetFlag: "-A assets_a",
496 },
497 {
498 name: "lib2",
499 },
500 {
501 name: "lib3",
502 assetPackages: []string{
503 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
504 buildDir + "/.intermediates/lib4/android_common/assets.zip",
505 },
506 },
507 {
508 name: "lib4",
509 assetFlag: "-A assets_b",
510 },
511 }
512 ctx := testApp(t, bp)
513
514 for _, test := range testCases {
515 t.Run(test.name, func(t *testing.T) {
516 m := ctx.ModuleForTests(test.name, "android_common")
517
518 // Check asset flag in aapt2 link flags
519 var aapt2link android.TestingBuildParams
520 if len(test.assetPackages) > 0 {
521 aapt2link = m.Output("aapt2/package-res.apk")
522 } else {
523 aapt2link = m.Output("package-res.apk")
524 }
525 aapt2Flags := aapt2link.Args["flags"]
526 if test.assetFlag != "" {
527 if !strings.Contains(aapt2Flags, test.assetFlag) {
528 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
529 }
530 } else {
531 if strings.Contains(aapt2Flags, " -A ") {
532 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
533 }
534 }
535
536 // Check asset merge rule.
537 if len(test.assetPackages) > 0 {
538 mergeAssets := m.Output("package-res.apk")
539 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
540 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
541 mergeAssets.Inputs.Strings(), test.assetPackages)
542 }
543 }
544 })
545 }
546}
547
Colin Crossbec85302019-02-13 13:15:46 -0800548func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800549 testCases := []struct {
550 name string
551 enforceRROTargets []string
552 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800553 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800554 overlayFiles map[string][]string
555 rroDirs map[string][]string
556 }{
557 {
558 name: "no RRO",
559 enforceRROTargets: nil,
560 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800561 resourceFiles: map[string][]string{
562 "foo": nil,
563 "bar": {"bar/res/res/values/strings.xml"},
564 "lib": nil,
565 "lib2": {"lib2/res/res/values/strings.xml"},
566 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800567 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800568 "foo": {
569 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800570 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000571 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800572 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800573 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
574 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000575 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800576 },
Colin Crossbec85302019-02-13 13:15:46 -0800577 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800578 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
579 "device/vendor/blah/overlay/bar/res/values/strings.xml",
580 },
Colin Crossbec85302019-02-13 13:15:46 -0800581 "lib": {
582 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
583 "lib/res/res/values/strings.xml",
584 "device/vendor/blah/overlay/lib/res/values/strings.xml",
585 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800586 },
587 rroDirs: map[string][]string{
588 "foo": nil,
589 "bar": nil,
590 },
591 },
592 {
593 name: "enforce RRO on foo",
594 enforceRROTargets: []string{"foo"},
595 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800596 resourceFiles: map[string][]string{
597 "foo": nil,
598 "bar": {"bar/res/res/values/strings.xml"},
599 "lib": nil,
600 "lib2": {"lib2/res/res/values/strings.xml"},
601 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800602 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800603 "foo": {
604 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800605 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000606 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800607 "foo/res/res/values/strings.xml",
608 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
609 },
Colin Crossbec85302019-02-13 13:15:46 -0800610 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800611 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
612 "device/vendor/blah/overlay/bar/res/values/strings.xml",
613 },
Colin Crossbec85302019-02-13 13:15:46 -0800614 "lib": {
615 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
616 "lib/res/res/values/strings.xml",
617 "device/vendor/blah/overlay/lib/res/values/strings.xml",
618 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800619 },
Colin Crossc1c37552019-01-31 11:42:41 -0800620
Colin Cross5c4791c2019-02-01 11:44:44 -0800621 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800622 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000623 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800624 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
625 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000626 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800627 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800628 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800629 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800630 },
631 },
632 {
633 name: "enforce RRO on all",
634 enforceRROTargets: []string{"*"},
635 enforceRROExcludedOverlays: []string{
636 // Excluding specific apps/res directories also allowed.
637 "device/vendor/blah/static_overlay/foo",
638 "device/vendor/blah/static_overlay/bar/res",
639 },
Colin Crossbec85302019-02-13 13:15:46 -0800640 resourceFiles: map[string][]string{
641 "foo": nil,
642 "bar": {"bar/res/res/values/strings.xml"},
643 "lib": nil,
644 "lib2": {"lib2/res/res/values/strings.xml"},
645 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800646 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800647 "foo": {
648 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800649 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000650 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800651 "foo/res/res/values/strings.xml",
652 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
653 },
Colin Crossbec85302019-02-13 13:15:46 -0800654 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
655 "lib": {
656 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
657 "lib/res/res/values/strings.xml",
658 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800659 },
660 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800661 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000662 "device:device/vendor/blah/overlay/foo/res",
663 "product:product/vendor/blah/overlay/foo/res",
664 // Lib dep comes after the direct deps
665 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800666 },
Anton Hansson53c88442019-03-18 15:53:16 +0000667 "bar": {"device:device/vendor/blah/overlay/bar/res"},
668 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800669 },
670 },
671 }
672
Anton Hansson53c88442019-03-18 15:53:16 +0000673 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800674 "device/vendor/blah/overlay",
675 "device/vendor/blah/overlay2",
676 "device/vendor/blah/static_overlay",
677 }
678
Anton Hansson53c88442019-03-18 15:53:16 +0000679 productResourceOverlays := []string{
680 "product/vendor/blah/overlay",
681 }
682
Colin Cross890ff552017-11-30 20:13:19 -0800683 fs := map[string][]byte{
684 "foo/res/res/values/strings.xml": nil,
685 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800686 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800687 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800688 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
689 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800690 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800691 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
692 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
693 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000694 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800695 }
696
697 bp := `
698 android_app {
699 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900700 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800701 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000702 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800703 }
704
705 android_app {
706 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900707 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800708 resource_dirs: ["bar/res"],
709 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800710
711 android_library {
712 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900713 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800714 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800715 static_libs: ["lib2"],
716 }
717
718 android_library {
719 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900720 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800721 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800722 }
Anton Hansson53c88442019-03-18 15:53:16 +0000723
724 // This library has the same resources as lib (should not lead to dupe RROs)
725 android_library {
726 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900727 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000728 resource_dirs: ["lib/res"]
729 }
Colin Cross890ff552017-11-30 20:13:19 -0800730 `
731
Colin Cross5c4791c2019-02-01 11:44:44 -0800732 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800733 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800734 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000735 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
736 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800737 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800738 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800739 }
740 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800741 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800742 }
743
Colin Cross98be1bb2019-12-13 20:41:13 -0800744 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800745 run(t, ctx, config)
746
Colin Crossbec85302019-02-13 13:15:46 -0800747 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
748 for _, o := range list {
749 res := module.MaybeOutput(o)
750 if res.Rule != nil {
751 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
752 // verify the inputs to the .arsc.flat rule.
753 files = append(files, res.Inputs.Strings()...)
754 } else {
755 // Otherwise, verify the full path to the output of the other module
756 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000757 }
Colin Cross890ff552017-11-30 20:13:19 -0800758 }
Colin Crossbec85302019-02-13 13:15:46 -0800759 return files
Colin Cross890ff552017-11-30 20:13:19 -0800760 }
761
Colin Crossbec85302019-02-13 13:15:46 -0800762 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
763 module := ctx.ModuleForTests(moduleName, "android_common")
764 resourceList := module.MaybeOutput("aapt2/res.list")
765 if resourceList.Rule != nil {
766 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000767 }
Colin Crossbec85302019-02-13 13:15:46 -0800768 overlayList := module.MaybeOutput("aapt2/overlay.list")
769 if overlayList.Rule != nil {
770 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
771 }
772
Anton Hansson53c88442019-03-18 15:53:16 +0000773 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
774 var prefix string
775 if d.overlayType == device {
776 prefix = "device:"
777 } else if d.overlayType == product {
778 prefix = "product:"
779 } else {
780 t.Fatalf("Unexpected overlayType %d", d.overlayType)
781 }
782 rroDirs = append(rroDirs, prefix+d.path.String())
783 }
Colin Crossbec85302019-02-13 13:15:46 -0800784
785 return resourceFiles, overlayFiles, rroDirs
786 }
787
788 modules := []string{"foo", "bar", "lib", "lib2"}
789 for _, module := range modules {
790 resourceFiles, overlayFiles, rroDirs := getResources(module)
791
792 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
793 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
794 module, testCase.resourceFiles[module], resourceFiles)
795 }
796 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
797 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
798 module, testCase.overlayFiles[module], overlayFiles)
799 }
800 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +0000801 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -0800802 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +0000803 }
Colin Cross890ff552017-11-30 20:13:19 -0800804 }
Colin Cross890ff552017-11-30 20:13:19 -0800805 })
806 }
807}
Colin Crossd09b0b62018-04-18 11:06:47 -0700808
809func TestAppSdkVersion(t *testing.T) {
810 testCases := []struct {
811 name string
812 sdkVersion string
813 platformSdkInt int
814 platformSdkCodename string
815 platformSdkFinal bool
816 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +0900817 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -0700818 }{
819 {
820 name: "current final SDK",
821 sdkVersion: "current",
822 platformSdkInt: 27,
823 platformSdkCodename: "REL",
824 platformSdkFinal: true,
825 expectedMinSdkVersion: "27",
826 },
827 {
828 name: "current non-final SDK",
829 sdkVersion: "current",
830 platformSdkInt: 27,
831 platformSdkCodename: "OMR1",
832 platformSdkFinal: false,
833 expectedMinSdkVersion: "OMR1",
834 },
835 {
836 name: "default final SDK",
837 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900838 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700839 platformSdkInt: 27,
840 platformSdkCodename: "REL",
841 platformSdkFinal: true,
842 expectedMinSdkVersion: "27",
843 },
844 {
845 name: "default non-final SDK",
846 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900847 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700848 platformSdkInt: 27,
849 platformSdkCodename: "OMR1",
850 platformSdkFinal: false,
851 expectedMinSdkVersion: "OMR1",
852 },
853 {
854 name: "14",
855 sdkVersion: "14",
856 expectedMinSdkVersion: "14",
857 },
858 }
859
860 for _, moduleType := range []string{"android_app", "android_library"} {
861 for _, test := range testCases {
862 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +0900863 platformApiProp := ""
864 if test.platformApis {
865 platformApiProp = "platform_apis: true,"
866 }
Colin Crossd09b0b62018-04-18 11:06:47 -0700867 bp := fmt.Sprintf(`%s {
868 name: "foo",
869 srcs: ["a.java"],
870 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900871 %s
872 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -0700873
Colin Cross98be1bb2019-12-13 20:41:13 -0800874 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -0700875 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
876 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
877 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
878
Colin Cross98be1bb2019-12-13 20:41:13 -0800879 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -0700880
881 run(t, ctx, config)
882
883 foo := ctx.ModuleForTests("foo", "android_common")
884 link := foo.Output("package-res.apk")
885 linkFlags := strings.Split(link.Args["flags"], " ")
886 min := android.IndexList("--min-sdk-version", linkFlags)
887 target := android.IndexList("--target-sdk-version", linkFlags)
888
889 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
890 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
891 }
892
893 gotMinSdkVersion := linkFlags[min+1]
894 gotTargetSdkVersion := linkFlags[target+1]
895
896 if gotMinSdkVersion != test.expectedMinSdkVersion {
897 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
898 test.expectedMinSdkVersion, gotMinSdkVersion)
899 }
900
901 if gotTargetSdkVersion != test.expectedMinSdkVersion {
902 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
903 test.expectedMinSdkVersion, gotTargetSdkVersion)
904 }
905 })
906 }
907 }
908}
Colin Crossa4f08812018-10-02 22:03:40 -0700909
Paul Duffin50c217c2019-06-12 13:25:22 +0100910func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700911 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +0100912 cc_library {
913 name: "libjni",
914 system_shared_libs: [],
Colin Cross01fd7cc2020-02-19 16:54:04 -0800915 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100916 stl: "none",
917 }
918
919 android_test {
920 name: "test",
921 sdk_version: "core_platform",
922 jni_libs: ["libjni"],
923 }
924
925 android_test {
926 name: "test_first",
927 sdk_version: "core_platform",
928 compile_multilib: "first",
929 jni_libs: ["libjni"],
930 }
931
932 android_test {
933 name: "test_both",
934 sdk_version: "core_platform",
935 compile_multilib: "both",
936 jni_libs: ["libjni"],
937 }
938
939 android_test {
940 name: "test_32",
941 sdk_version: "core_platform",
942 compile_multilib: "32",
943 jni_libs: ["libjni"],
944 }
945
946 android_test {
947 name: "test_64",
948 sdk_version: "core_platform",
949 compile_multilib: "64",
950 jni_libs: ["libjni"],
951 }
952 `)
953
954 testCases := []struct {
955 name string
956 abis []string
957 }{
958 {"test", []string{"arm64-v8a"}},
959 {"test_first", []string{"arm64-v8a"}},
960 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
961 {"test_32", []string{"armeabi-v7a"}},
962 {"test_64", []string{"arm64-v8a"}},
963 }
964
965 for _, test := range testCases {
966 t.Run(test.name, func(t *testing.T) {
967 app := ctx.ModuleForTests(test.name, "android_common")
968 jniLibZip := app.Output("jnilibs.zip")
969 var abis []string
970 args := strings.Fields(jniLibZip.Args["jarArgs"])
971 for i := 0; i < len(args); i++ {
972 if args[i] == "-P" {
973 abis = append(abis, filepath.Base(args[i+1]))
974 i++
975 }
976 }
977 if !reflect.DeepEqual(abis, test.abis) {
978 t.Errorf("want abis %v, got %v", test.abis, abis)
979 }
980 })
981 }
982}
983
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900984func TestAppSdkVersionByPartition(t *testing.T) {
985 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
986 android_app {
987 name: "foo",
988 srcs: ["a.java"],
989 vendor: true,
990 platform_apis: true,
991 }
992 `)
993
994 testJava(t, `
995 android_app {
996 name: "bar",
997 srcs: ["b.java"],
998 platform_apis: true,
999 }
1000 `)
1001
1002 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001003 bp := `
1004 android_app {
1005 name: "foo",
1006 srcs: ["a.java"],
1007 product_specific: true,
1008 platform_apis: true,
1009 }
1010 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001011
1012 config := testAppConfig(nil, bp, nil)
1013 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001014 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -08001015 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 +09001016 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -08001017 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001018 }
1019 }
1020}
1021
Paul Duffin50c217c2019-06-12 13:25:22 +01001022func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001023 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001024 cc_library {
1025 name: "libjni",
1026 system_shared_libs: [],
1027 stl: "none",
Colin Cross1c93c292020-02-15 10:38:00 -08001028 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001029 }
1030
1031 android_app {
1032 name: "app",
1033 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001034 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001035 }
1036
1037 android_app {
1038 name: "app_noembed",
1039 jni_libs: ["libjni"],
1040 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001041 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001042 }
1043
1044 android_app {
1045 name: "app_embed",
1046 jni_libs: ["libjni"],
1047 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001048 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001049 }
1050
1051 android_test {
1052 name: "test",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001053 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001054 jni_libs: ["libjni"],
1055 }
1056
1057 android_test {
1058 name: "test_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001059 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001060 jni_libs: ["libjni"],
1061 use_embedded_native_libs: false,
1062 }
1063
1064 android_test_helper_app {
1065 name: "test_helper",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001066 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001067 jni_libs: ["libjni"],
1068 }
1069
1070 android_test_helper_app {
1071 name: "test_helper_noembed",
Colin Cross01fd7cc2020-02-19 16:54:04 -08001072 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001073 jni_libs: ["libjni"],
1074 use_embedded_native_libs: false,
1075 }
1076 `)
1077
1078 testCases := []struct {
1079 name string
1080 packaged bool
1081 compressed bool
1082 }{
1083 {"app", false, false},
1084 {"app_noembed", false, false},
1085 {"app_embed", true, false},
1086 {"test", true, false},
1087 {"test_noembed", true, true},
1088 {"test_helper", true, false},
1089 {"test_helper_noembed", true, true},
1090 }
1091
1092 for _, test := range testCases {
1093 t.Run(test.name, func(t *testing.T) {
1094 app := ctx.ModuleForTests(test.name, "android_common")
1095 jniLibZip := app.MaybeOutput("jnilibs.zip")
1096 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1097 t.Errorf("expected jni packaged %v, got %v", w, g)
1098 }
1099
1100 if jniLibZip.Rule != nil {
1101 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1102 t.Errorf("expected jni compressed %v, got %v", w, g)
1103 }
Colin Cross01fd7cc2020-02-19 16:54:04 -08001104
1105 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
1106 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
1107 }
Paul Duffin50c217c2019-06-12 13:25:22 +01001108 }
1109 })
1110 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001111}
1112
Colin Cross1dd9c442020-05-08 11:20:24 -07001113func TestJNISDK(t *testing.T) {
1114 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
1115 cc_library {
1116 name: "libjni",
1117 system_shared_libs: [],
1118 stl: "none",
1119 sdk_version: "current",
1120 }
1121
1122 android_test {
1123 name: "app_platform",
1124 jni_libs: ["libjni"],
1125 platform_apis: true,
1126 }
1127
1128 android_test {
1129 name: "app_sdk",
1130 jni_libs: ["libjni"],
1131 sdk_version: "current",
1132 }
1133
1134 android_test {
1135 name: "app_force_platform",
1136 jni_libs: ["libjni"],
1137 sdk_version: "current",
1138 jni_uses_platform_apis: true,
1139 }
1140
1141 android_test {
1142 name: "app_force_sdk",
1143 jni_libs: ["libjni"],
1144 platform_apis: true,
1145 jni_uses_sdk_apis: true,
1146 }
Colin Crosseb032962020-05-13 11:05:02 -07001147
1148 cc_library {
1149 name: "libvendorjni",
1150 system_shared_libs: [],
1151 stl: "none",
1152 vendor: true,
1153 }
1154
1155 android_test {
1156 name: "app_vendor",
1157 jni_libs: ["libvendorjni"],
1158 sdk_version: "current",
1159 vendor: true,
1160 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001161 `)
1162
1163 testCases := []struct {
Colin Crosseb032962020-05-13 11:05:02 -07001164 name string
1165 sdkJNI bool
1166 vendorJNI bool
Colin Cross1dd9c442020-05-08 11:20:24 -07001167 }{
Colin Crosseb032962020-05-13 11:05:02 -07001168 {name: "app_platform"},
1169 {name: "app_sdk", sdkJNI: true},
1170 {name: "app_force_platform"},
1171 {name: "app_force_sdk", sdkJNI: true},
1172 {name: "app_vendor", vendorJNI: true},
Colin Cross1dd9c442020-05-08 11:20:24 -07001173 }
1174
Colin Crosseb032962020-05-13 11:05:02 -07001175 platformJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_shared").
1176 Output("libjni.so").Output.String()
1177 sdkJNI := ctx.ModuleForTests("libjni", "android_arm64_armv8-a_sdk_shared").
1178 Output("libjni.so").Output.String()
1179 vendorJNI := ctx.ModuleForTests("libvendorjni", "android_arm64_armv8-a_shared").
1180 Output("libvendorjni.so").Output.String()
1181
Colin Cross1dd9c442020-05-08 11:20:24 -07001182 for _, test := range testCases {
1183 t.Run(test.name, func(t *testing.T) {
1184 app := ctx.ModuleForTests(test.name, "android_common")
Colin Cross1dd9c442020-05-08 11:20:24 -07001185
1186 jniLibZip := app.MaybeOutput("jnilibs.zip")
1187 if len(jniLibZip.Implicits) != 1 {
1188 t.Fatalf("expected exactly one jni library, got %q", jniLibZip.Implicits.Strings())
1189 }
1190 gotJNI := jniLibZip.Implicits[0].String()
1191
1192 if test.sdkJNI {
1193 if gotJNI != sdkJNI {
1194 t.Errorf("expected SDK JNI library %q, got %q", sdkJNI, gotJNI)
1195 }
Colin Crosseb032962020-05-13 11:05:02 -07001196 } else if test.vendorJNI {
1197 if gotJNI != vendorJNI {
1198 t.Errorf("expected platform JNI library %q, got %q", vendorJNI, gotJNI)
1199 }
Colin Cross1dd9c442020-05-08 11:20:24 -07001200 } else {
1201 if gotJNI != platformJNI {
1202 t.Errorf("expected platform JNI library %q, got %q", platformJNI, gotJNI)
1203 }
1204 }
1205 })
1206 }
1207
1208 t.Run("jni_uses_platform_apis_error", func(t *testing.T) {
1209 testJavaError(t, `jni_uses_platform_apis: can only be set for modules that set sdk_version`, `
1210 android_test {
1211 name: "app_platform",
1212 platform_apis: true,
1213 jni_uses_platform_apis: true,
1214 }
1215 `)
1216 })
1217
1218 t.Run("jni_uses_sdk_apis_error", func(t *testing.T) {
1219 testJavaError(t, `jni_uses_sdk_apis: can only be set for modules that do not set sdk_version`, `
1220 android_test {
1221 name: "app_sdk",
1222 sdk_version: "current",
1223 jni_uses_sdk_apis: true,
1224 }
1225 `)
1226 })
1227
1228}
1229
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001230func TestCertificates(t *testing.T) {
1231 testCases := []struct {
1232 name string
1233 bp string
1234 certificateOverride string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001235 expectedLineage string
1236 expectedCertificate string
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001237 }{
1238 {
1239 name: "default",
1240 bp: `
1241 android_app {
1242 name: "foo",
1243 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001244 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001245 }
1246 `,
1247 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001248 expectedLineage: "",
1249 expectedCertificate: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001250 },
1251 {
1252 name: "module certificate property",
1253 bp: `
1254 android_app {
1255 name: "foo",
1256 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001257 certificate: ":new_certificate",
1258 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001259 }
1260
1261 android_app_certificate {
1262 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001263 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001264 }
1265 `,
1266 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001267 expectedLineage: "",
1268 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001269 },
1270 {
1271 name: "path certificate property",
1272 bp: `
1273 android_app {
1274 name: "foo",
1275 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001276 certificate: "expiredkey",
1277 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001278 }
1279 `,
1280 certificateOverride: "",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001281 expectedLineage: "",
1282 expectedCertificate: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001283 },
1284 {
1285 name: "certificate overrides",
1286 bp: `
1287 android_app {
1288 name: "foo",
1289 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001290 certificate: "expiredkey",
1291 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001292 }
1293
1294 android_app_certificate {
1295 name: "new_certificate",
Colin Cross1dd9c442020-05-08 11:20:24 -07001296 certificate: "cert/new_cert",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001297 }
1298 `,
1299 certificateOverride: "foo:new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001300 expectedLineage: "",
1301 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
1302 },
1303 {
1304 name: "certificate lineage",
1305 bp: `
1306 android_app {
1307 name: "foo",
1308 srcs: ["a.java"],
1309 certificate: ":new_certificate",
1310 lineage: "lineage.bin",
1311 sdk_version: "current",
1312 }
1313
1314 android_app_certificate {
1315 name: "new_certificate",
1316 certificate: "cert/new_cert",
1317 }
1318 `,
1319 certificateOverride: "",
1320 expectedLineage: "--lineage lineage.bin",
1321 expectedCertificate: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001322 },
1323 }
1324
1325 for _, test := range testCases {
1326 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001327 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001328 if test.certificateOverride != "" {
1329 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1330 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001331 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001332
1333 run(t, ctx, config)
1334 foo := ctx.ModuleForTests("foo", "android_common")
1335
1336 signapk := foo.Output("foo.apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001337 signCertificateFlags := signapk.Args["certificates"]
1338 if test.expectedCertificate != signCertificateFlags {
1339 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedCertificate, signCertificateFlags)
1340 }
1341
1342 signFlags := signapk.Args["flags"]
1343 if test.expectedLineage != signFlags {
1344 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expectedLineage, signFlags)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001345 }
1346 })
1347 }
1348}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001349
Songchun Fan688de9a2020-03-24 20:32:24 -07001350func TestRequestV4SigningFlag(t *testing.T) {
1351 testCases := []struct {
1352 name string
1353 bp string
1354 expected string
1355 }{
1356 {
1357 name: "default",
1358 bp: `
1359 android_app {
1360 name: "foo",
1361 srcs: ["a.java"],
1362 sdk_version: "current",
1363 }
1364 `,
1365 expected: "",
1366 },
1367 {
1368 name: "default",
1369 bp: `
1370 android_app {
1371 name: "foo",
1372 srcs: ["a.java"],
1373 sdk_version: "current",
1374 v4_signature: false,
1375 }
1376 `,
1377 expected: "",
1378 },
1379 {
1380 name: "module certificate property",
1381 bp: `
1382 android_app {
1383 name: "foo",
1384 srcs: ["a.java"],
1385 sdk_version: "current",
1386 v4_signature: true,
1387 }
1388 `,
1389 expected: "--enable-v4",
1390 },
1391 }
1392
1393 for _, test := range testCases {
1394 t.Run(test.name, func(t *testing.T) {
1395 config := testAppConfig(nil, test.bp, nil)
1396 ctx := testContext()
1397
1398 run(t, ctx, config)
1399 foo := ctx.ModuleForTests("foo", "android_common")
1400
1401 signapk := foo.Output("foo.apk")
1402 signFlags := signapk.Args["flags"]
1403 if test.expected != signFlags {
1404 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1405 }
1406 })
1407 }
1408}
1409
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001410func TestPackageNameOverride(t *testing.T) {
1411 testCases := []struct {
1412 name string
1413 bp string
1414 packageNameOverride string
1415 expected []string
1416 }{
1417 {
1418 name: "default",
1419 bp: `
1420 android_app {
1421 name: "foo",
1422 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001423 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001424 }
1425 `,
1426 packageNameOverride: "",
1427 expected: []string{
1428 buildDir + "/.intermediates/foo/android_common/foo.apk",
1429 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1430 },
1431 },
1432 {
1433 name: "overridden",
1434 bp: `
1435 android_app {
1436 name: "foo",
1437 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001438 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001439 }
1440 `,
1441 packageNameOverride: "foo:bar",
1442 expected: []string{
1443 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001444 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001445 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1446 },
1447 },
1448 }
1449
1450 for _, test := range testCases {
1451 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001452 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001453 if test.packageNameOverride != "" {
1454 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1455 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001456 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001457
1458 run(t, ctx, config)
1459 foo := ctx.ModuleForTests("foo", "android_common")
1460
1461 outputs := foo.AllOutputs()
1462 outputMap := make(map[string]bool)
1463 for _, o := range outputs {
1464 outputMap[o] = true
1465 }
1466 for _, e := range test.expected {
1467 if _, exist := outputMap[e]; !exist {
1468 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1469 }
1470 }
1471 })
1472 }
1473}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001474
1475func TestInstrumentationTargetOverridden(t *testing.T) {
1476 bp := `
1477 android_app {
1478 name: "foo",
1479 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001480 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001481 }
1482
1483 android_test {
1484 name: "bar",
1485 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001486 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001487 }
1488 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001489 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001490 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001491 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001492
1493 run(t, ctx, config)
1494
1495 bar := ctx.ModuleForTests("bar", "android_common")
1496 res := bar.Output("package-res.apk")
1497 aapt2Flags := res.Args["flags"]
1498 e := "--rename-instrumentation-target-package org.dandroid.bp"
1499 if !strings.Contains(aapt2Flags, e) {
1500 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1501 }
1502}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001503
1504func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001505 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001506 android_app {
1507 name: "foo",
1508 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001509 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001510 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001511 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001512 }
1513
1514 override_android_app {
1515 name: "bar",
1516 base: "foo",
1517 certificate: ":new_certificate",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001518 lineage: "lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001519 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001520 }
1521
1522 android_app_certificate {
1523 name: "new_certificate",
1524 certificate: "cert/new_cert",
1525 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001526
1527 override_android_app {
1528 name: "baz",
1529 base: "foo",
1530 package_name: "org.dandroid.bp",
1531 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001532 `)
1533
1534 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001535 moduleName string
1536 variantName string
1537 apkName string
1538 apkPath string
Liz Kammer70dd74d2020-05-07 13:24:05 -07001539 certFlag string
1540 lineageFlag string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001541 overrides []string
1542 aaptFlag string
1543 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001544 }{
1545 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001546 moduleName: "foo",
1547 variantName: "android_common",
1548 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001549 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1550 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001551 overrides: []string{"qux"},
1552 aaptFlag: "",
1553 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001554 },
1555 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001556 moduleName: "bar",
1557 variantName: "android_common_bar",
1558 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001559 certFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1560 lineageFlag: "--lineage lineage.bin",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001561 overrides: []string{"qux", "foo"},
1562 aaptFlag: "",
1563 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001564 },
1565 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001566 moduleName: "baz",
1567 variantName: "android_common_baz",
1568 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Liz Kammer70dd74d2020-05-07 13:24:05 -07001569 certFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1570 lineageFlag: "",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001571 overrides: []string{"qux", "foo"},
1572 aaptFlag: "--rename-manifest-package org.dandroid.bp",
1573 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001574 },
1575 }
1576 for _, expected := range expectedVariants {
1577 variant := ctx.ModuleForTests("foo", expected.variantName)
1578
1579 // Check the final apk name
1580 outputs := variant.AllOutputs()
1581 expectedApkPath := buildDir + expected.apkPath
1582 found := false
1583 for _, o := range outputs {
1584 if o == expectedApkPath {
1585 found = true
1586 break
1587 }
1588 }
1589 if !found {
1590 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1591 }
1592
1593 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001594 signapk := variant.Output(expected.moduleName + ".apk")
Liz Kammer70dd74d2020-05-07 13:24:05 -07001595 certFlag := signapk.Args["certificates"]
1596 if expected.certFlag != certFlag {
1597 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.certFlag, certFlag)
1598 }
1599
1600 // Check the lineage flags
1601 lineageFlag := signapk.Args["flags"]
1602 if expected.lineageFlag != lineageFlag {
1603 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.lineageFlag, lineageFlag)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001604 }
1605
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001606 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001607 mod := variant.Module().(*AndroidApp)
1608 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1609 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1610 expected.overrides, mod.appProperties.Overrides)
1611 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001612
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001613 // Test Overridable property: Logging_parent
1614 logging_parent := mod.aapt.LoggingParent
1615 if expected.logging_parent != logging_parent {
1616 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1617 expected.logging_parent, logging_parent)
1618 }
1619
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001620 // Check the package renaming flag, if exists.
1621 res := variant.Output("package-res.apk")
1622 aapt2Flags := res.Args["flags"]
1623 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1624 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1625 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001626 }
1627}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001628
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001629func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001630 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001631 android_app {
1632 name: "foo",
1633 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001634 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001635 }
1636
1637 override_android_app {
1638 name: "bar",
1639 base: "foo",
1640 package_name: "org.dandroid.bp",
1641 }
1642
1643 android_test {
1644 name: "baz",
1645 srcs: ["b.java"],
1646 instrumentation_for: "foo",
1647 }
1648
1649 android_test {
1650 name: "qux",
1651 srcs: ["b.java"],
1652 instrumentation_for: "bar",
1653 }
1654 `)
1655
1656 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1657 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1658 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1659 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1660 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1661 }
1662
1663 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1664 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1665 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1666 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1667 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1668 }
1669}
1670
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001671func TestOverrideAndroidTest(t *testing.T) {
1672 ctx, _ := testJava(t, `
1673 android_app {
1674 name: "foo",
1675 srcs: ["a.java"],
1676 package_name: "com.android.foo",
1677 sdk_version: "current",
1678 }
1679
1680 override_android_app {
1681 name: "bar",
1682 base: "foo",
1683 package_name: "com.android.bar",
1684 }
1685
1686 android_test {
1687 name: "foo_test",
1688 srcs: ["b.java"],
1689 instrumentation_for: "foo",
1690 }
1691
1692 override_android_test {
1693 name: "bar_test",
1694 base: "foo_test",
1695 package_name: "com.android.bar.test",
1696 instrumentation_for: "bar",
1697 instrumentation_target_package: "com.android.bar",
1698 }
1699 `)
1700
1701 expectedVariants := []struct {
1702 moduleName string
1703 variantName string
1704 apkPath string
1705 overrides []string
1706 targetVariant string
1707 packageFlag string
1708 targetPackageFlag string
1709 }{
1710 {
1711 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001712 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001713 overrides: nil,
1714 targetVariant: "android_common",
1715 packageFlag: "",
1716 targetPackageFlag: "",
1717 },
1718 {
1719 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001720 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001721 overrides: []string{"foo_test"},
1722 targetVariant: "android_common_bar",
1723 packageFlag: "com.android.bar.test",
1724 targetPackageFlag: "com.android.bar",
1725 },
1726 }
1727 for _, expected := range expectedVariants {
1728 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1729
1730 // Check the final apk name
1731 outputs := variant.AllOutputs()
1732 expectedApkPath := buildDir + expected.apkPath
1733 found := false
1734 for _, o := range outputs {
1735 if o == expectedApkPath {
1736 found = true
1737 break
1738 }
1739 }
1740 if !found {
1741 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1742 }
1743
1744 // Check if the overrides field values are correctly aggregated.
1745 mod := variant.Module().(*AndroidTest)
1746 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1747 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1748 expected.overrides, mod.appProperties.Overrides)
1749 }
1750
1751 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1752 javac := variant.Rule("javac")
1753 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1754 if !strings.Contains(javac.Args["classpath"], turbine) {
1755 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1756 }
1757
1758 // Check aapt2 flags.
1759 res := variant.Output("package-res.apk")
1760 aapt2Flags := res.Args["flags"]
1761 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1762 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1763 }
1764}
1765
Jaewoong Jung39982342020-01-14 10:27:18 -08001766func TestAndroidTest_FixTestConfig(t *testing.T) {
1767 ctx, _ := testJava(t, `
1768 android_app {
1769 name: "foo",
1770 srcs: ["a.java"],
1771 package_name: "com.android.foo",
1772 sdk_version: "current",
1773 }
1774
1775 android_test {
1776 name: "foo_test",
1777 srcs: ["b.java"],
1778 instrumentation_for: "foo",
1779 }
1780
1781 android_test {
1782 name: "bar_test",
1783 srcs: ["b.java"],
1784 package_name: "com.android.bar.test",
1785 instrumentation_for: "foo",
1786 }
1787
1788 override_android_test {
1789 name: "baz_test",
1790 base: "foo_test",
1791 package_name: "com.android.baz.test",
1792 }
1793 `)
1794
1795 testCases := []struct {
1796 moduleName string
1797 variantName string
1798 expectedFlags []string
1799 }{
1800 {
1801 moduleName: "foo_test",
1802 variantName: "android_common",
1803 },
1804 {
1805 moduleName: "bar_test",
1806 variantName: "android_common",
1807 expectedFlags: []string{
1808 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
1809 "--package-name com.android.bar.test",
1810 },
1811 },
1812 {
1813 moduleName: "foo_test",
1814 variantName: "android_common_baz_test",
1815 expectedFlags: []string{
1816 "--manifest " + buildDir +
1817 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
1818 "--package-name com.android.baz.test",
1819 "--test-file-name baz_test.apk",
1820 },
1821 },
1822 }
1823
1824 for _, test := range testCases {
1825 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
1826 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
1827
1828 if len(test.expectedFlags) > 0 {
1829 if params.Rule == nil {
1830 t.Errorf("test_config_fixer was expected to run, but didn't")
1831 } else {
1832 for _, flag := range test.expectedFlags {
1833 if !strings.Contains(params.RuleParams.Command, flag) {
1834 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
1835 }
1836 }
1837 }
1838 } else {
1839 if params.Rule != nil {
1840 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
1841 }
1842 }
1843
1844 }
1845}
1846
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001847func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001848 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001849 android_app_import {
1850 name: "foo",
1851 apk: "prebuilts/apk/app.apk",
1852 certificate: "platform",
1853 dex_preopt: {
1854 enabled: true,
1855 },
1856 }
1857 `)
1858
1859 variant := ctx.ModuleForTests("foo", "android_common")
1860
1861 // Check dexpreopt outputs.
1862 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1863 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1864 t.Errorf("can't find dexpreopt outputs")
1865 }
1866
1867 // Check cert signing flag.
1868 signedApk := variant.Output("signed/foo.apk")
1869 signingFlag := signedApk.Args["certificates"]
1870 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
1871 if expected != signingFlag {
1872 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1873 }
1874}
1875
1876func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001877 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001878 android_app_import {
1879 name: "foo",
1880 apk: "prebuilts/apk/app.apk",
1881 certificate: "platform",
1882 dex_preopt: {
1883 enabled: false,
1884 },
1885 }
1886 `)
1887
1888 variant := ctx.ModuleForTests("foo", "android_common")
1889
1890 // Check dexpreopt outputs. They shouldn't exist.
1891 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
1892 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
1893 t.Errorf("dexpreopt shouldn't have run.")
1894 }
1895}
1896
1897func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001898 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001899 android_app_import {
1900 name: "foo",
1901 apk: "prebuilts/apk/app.apk",
1902 presigned: true,
1903 dex_preopt: {
1904 enabled: true,
1905 },
1906 }
1907 `)
1908
1909 variant := ctx.ModuleForTests("foo", "android_common")
1910
1911 // Check dexpreopt outputs.
1912 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1913 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1914 t.Errorf("can't find dexpreopt outputs")
1915 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01001916 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001917 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
1918 t.Errorf("signing rule shouldn't be included.")
1919 }
1920 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
1921 t.Errorf("can't find aligning rule")
1922 }
1923}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001924
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001925func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
1926 ctx, _ := testJava(t, `
1927 android_app_import {
1928 name: "foo",
1929 apk: "prebuilts/apk/app.apk",
1930 default_dev_cert: true,
1931 dex_preopt: {
1932 enabled: true,
1933 },
1934 }
1935 `)
1936
1937 variant := ctx.ModuleForTests("foo", "android_common")
1938
1939 // Check dexpreopt outputs.
1940 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1941 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1942 t.Errorf("can't find dexpreopt outputs")
1943 }
1944
1945 // Check cert signing flag.
1946 signedApk := variant.Output("signed/foo.apk")
1947 signingFlag := signedApk.Args["certificates"]
1948 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
1949 if expected != signingFlag {
1950 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1951 }
1952}
1953
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001954func TestAndroidAppImport_DpiVariants(t *testing.T) {
1955 bp := `
1956 android_app_import {
1957 name: "foo",
1958 apk: "prebuilts/apk/app.apk",
1959 dpi_variants: {
1960 xhdpi: {
1961 apk: "prebuilts/apk/app_xhdpi.apk",
1962 },
1963 xxhdpi: {
1964 apk: "prebuilts/apk/app_xxhdpi.apk",
1965 },
1966 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001967 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001968 dex_preopt: {
1969 enabled: true,
1970 },
1971 }
1972 `
1973 testCases := []struct {
1974 name string
1975 aaptPreferredConfig *string
1976 aaptPrebuiltDPI []string
1977 expected string
1978 }{
1979 {
1980 name: "no preferred",
1981 aaptPreferredConfig: nil,
1982 aaptPrebuiltDPI: []string{},
1983 expected: "prebuilts/apk/app.apk",
1984 },
1985 {
1986 name: "AAPTPreferredConfig matches",
1987 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001988 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001989 expected: "prebuilts/apk/app_xhdpi.apk",
1990 },
1991 {
1992 name: "AAPTPrebuiltDPI matches",
1993 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1994 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
1995 expected: "prebuilts/apk/app_xxhdpi.apk",
1996 },
1997 {
1998 name: "non-first AAPTPrebuiltDPI matches",
1999 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2000 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
2001 expected: "prebuilts/apk/app_xhdpi.apk",
2002 },
2003 {
2004 name: "no matches",
2005 aaptPreferredConfig: proptools.StringPtr("mdpi"),
2006 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
2007 expected: "prebuilts/apk/app.apk",
2008 },
2009 }
2010
2011 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2012 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08002013 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002014 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
2015 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08002016 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07002017
2018 run(t, ctx, config)
2019
2020 variant := ctx.ModuleForTests("foo", "android_common")
2021 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2022 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2023 if len(matches) != 2 {
2024 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2025 }
2026 if test.expected != matches[1] {
2027 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2028 }
2029 }
2030}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002031
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002032func TestAndroidAppImport_Filename(t *testing.T) {
2033 ctx, config := testJava(t, `
2034 android_app_import {
2035 name: "foo",
2036 apk: "prebuilts/apk/app.apk",
2037 presigned: true,
2038 }
2039
2040 android_app_import {
2041 name: "bar",
2042 apk: "prebuilts/apk/app.apk",
2043 presigned: true,
2044 filename: "bar_sample.apk"
2045 }
2046 `)
2047
2048 testCases := []struct {
2049 name string
2050 expected string
2051 }{
2052 {
2053 name: "foo",
2054 expected: "foo.apk",
2055 },
2056 {
2057 name: "bar",
2058 expected: "bar_sample.apk",
2059 },
2060 }
2061
2062 for _, test := range testCases {
2063 variant := ctx.ModuleForTests(test.name, "android_common")
2064 if variant.MaybeOutput(test.expected).Rule == nil {
2065 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
2066 }
2067
2068 a := variant.Module().(*AndroidAppImport)
2069 expectedValues := []string{test.expected}
2070 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002071 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07002072 if !reflect.DeepEqual(actualValues, expectedValues) {
2073 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
2074 actualValues, expectedValues)
2075 }
2076 }
2077}
2078
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002079func TestAndroidAppImport_ArchVariants(t *testing.T) {
2080 // The test config's target arch is ARM64.
2081 testCases := []struct {
2082 name string
2083 bp string
2084 expected string
2085 }{
2086 {
2087 name: "matching arch",
2088 bp: `
2089 android_app_import {
2090 name: "foo",
2091 apk: "prebuilts/apk/app.apk",
2092 arch: {
2093 arm64: {
2094 apk: "prebuilts/apk/app_arm64.apk",
2095 },
2096 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002097 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002098 dex_preopt: {
2099 enabled: true,
2100 },
2101 }
2102 `,
2103 expected: "prebuilts/apk/app_arm64.apk",
2104 },
2105 {
2106 name: "no matching arch",
2107 bp: `
2108 android_app_import {
2109 name: "foo",
2110 apk: "prebuilts/apk/app.apk",
2111 arch: {
2112 arm: {
2113 apk: "prebuilts/apk/app_arm.apk",
2114 },
2115 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07002116 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07002117 dex_preopt: {
2118 enabled: true,
2119 },
2120 }
2121 `,
2122 expected: "prebuilts/apk/app.apk",
2123 },
2124 }
2125
2126 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
2127 for _, test := range testCases {
2128 ctx, _ := testJava(t, test.bp)
2129
2130 variant := ctx.ModuleForTests("foo", "android_common")
2131 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2132 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
2133 if len(matches) != 2 {
2134 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
2135 }
2136 if test.expected != matches[1] {
2137 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
2138 }
2139 }
2140}
2141
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002142func TestAndroidTestImport(t *testing.T) {
2143 ctx, config := testJava(t, `
2144 android_test_import {
2145 name: "foo",
2146 apk: "prebuilts/apk/app.apk",
2147 presigned: true,
2148 data: [
2149 "testdata/data",
2150 ],
2151 }
2152 `)
2153
2154 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
2155
2156 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09002157 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07002158 expected := []string{"tests"}
2159 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
2160 if !reflect.DeepEqual(expected, actual) {
2161 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
2162 }
2163 expected = []string{"testdata/data:testdata/data"}
2164 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
2165 if !reflect.DeepEqual(expected, actual) {
2166 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
2167 }
2168}
2169
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08002170func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
2171 ctx, _ := testJava(t, `
2172 android_test_import {
2173 name: "foo",
2174 apk: "prebuilts/apk/app.apk",
2175 certificate: "cert/new_cert",
2176 data: [
2177 "testdata/data",
2178 ],
2179 }
2180
2181 android_test_import {
2182 name: "foo_presigned",
2183 apk: "prebuilts/apk/app.apk",
2184 presigned: true,
2185 data: [
2186 "testdata/data",
2187 ],
2188 }
2189 `)
2190
2191 variant := ctx.ModuleForTests("foo", "android_common")
2192 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2193 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2194 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2195 }
2196
2197 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2198 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2199 if jniRule != android.Cp.String() {
2200 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2201 }
2202}
2203
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002204func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002205 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002206 cc_library {
2207 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002208 sdk_version: "current",
2209 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002210 }
2211
2212 android_test {
2213 name: "stl",
2214 jni_libs: ["libjni"],
2215 compile_multilib: "both",
2216 sdk_version: "current",
2217 stl: "c++_shared",
2218 }
2219
2220 android_test {
2221 name: "system",
2222 jni_libs: ["libjni"],
2223 compile_multilib: "both",
2224 sdk_version: "current",
2225 }
2226 `)
2227
2228 testCases := []struct {
2229 name string
2230 jnis []string
2231 }{
2232 {"stl",
2233 []string{
2234 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002235 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002236 },
2237 },
2238 {"system",
2239 []string{
2240 "libjni.so",
2241 },
2242 },
2243 }
2244
2245 for _, test := range testCases {
2246 t.Run(test.name, func(t *testing.T) {
2247 app := ctx.ModuleForTests(test.name, "android_common")
2248 jniLibZip := app.Output("jnilibs.zip")
2249 var jnis []string
2250 args := strings.Fields(jniLibZip.Args["jarArgs"])
2251 for i := 0; i < len(args); i++ {
2252 if args[i] == "-f" {
2253 jnis = append(jnis, args[i+1])
2254 i += 1
2255 }
2256 }
2257 jnisJoined := strings.Join(jnis, " ")
2258 for _, jni := range test.jnis {
2259 if !strings.Contains(jnisJoined, jni) {
2260 t.Errorf("missing jni %q in %q", jni, jnis)
2261 }
2262 }
2263 })
2264 }
2265}
Colin Cross50ddcc42019-05-16 12:28:22 -07002266
2267func TestUsesLibraries(t *testing.T) {
2268 bp := `
2269 java_sdk_library {
2270 name: "foo",
2271 srcs: ["a.java"],
2272 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002273 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002274 }
2275
2276 java_sdk_library {
2277 name: "bar",
2278 srcs: ["a.java"],
2279 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002280 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002281 }
2282
2283 android_app {
2284 name: "app",
2285 srcs: ["a.java"],
2286 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002287 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002288 optional_uses_libs: [
2289 "bar",
2290 "baz",
2291 ],
2292 }
2293
2294 android_app_import {
2295 name: "prebuilt",
2296 apk: "prebuilts/apk/app.apk",
2297 certificate: "platform",
2298 uses_libs: ["foo"],
2299 optional_uses_libs: [
2300 "bar",
2301 "baz",
2302 ],
2303 }
2304 `
2305
Colin Cross98be1bb2019-12-13 20:41:13 -08002306 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002307 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2308
Colin Cross98be1bb2019-12-13 20:41:13 -08002309 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002310
2311 run(t, ctx, config)
2312
2313 app := ctx.ModuleForTests("app", "android_common")
2314 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2315
2316 // Test that all libraries are verified
2317 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2318 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2319 t.Errorf("wanted %q in %q", w, cmd)
2320 }
2321
2322 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2323 t.Errorf("wanted %q in %q", w, cmd)
2324 }
2325
2326 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2327
2328 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2329 t.Errorf("wanted %q in %q", w, cmd)
2330 }
2331
2332 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2333 t.Errorf("wanted %q in %q", w, cmd)
2334 }
2335
2336 // Test that only present libraries are preopted
2337 cmd = app.Rule("dexpreopt").RuleParams.Command
2338
2339 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2340 t.Errorf("wanted %q in %q", w, cmd)
2341 }
2342
2343 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2344
2345 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2346 t.Errorf("wanted %q in %q", w, cmd)
2347 }
2348}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002349
2350func TestCodelessApp(t *testing.T) {
2351 testCases := []struct {
2352 name string
2353 bp string
2354 noCode bool
2355 }{
2356 {
2357 name: "normal",
2358 bp: `
2359 android_app {
2360 name: "foo",
2361 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002362 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002363 }
2364 `,
2365 noCode: false,
2366 },
2367 {
2368 name: "app without sources",
2369 bp: `
2370 android_app {
2371 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002372 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002373 }
2374 `,
2375 noCode: true,
2376 },
2377 {
2378 name: "app with libraries",
2379 bp: `
2380 android_app {
2381 name: "foo",
2382 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002383 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002384 }
2385
2386 java_library {
2387 name: "lib",
2388 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002389 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002390 }
2391 `,
2392 noCode: false,
2393 },
2394 {
2395 name: "app with sourceless libraries",
2396 bp: `
2397 android_app {
2398 name: "foo",
2399 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002400 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002401 }
2402
2403 java_library {
2404 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002405 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002406 }
2407 `,
2408 // TODO(jungjw): this should probably be true
2409 noCode: false,
2410 },
2411 }
2412
2413 for _, test := range testCases {
2414 t.Run(test.name, func(t *testing.T) {
2415 ctx := testApp(t, test.bp)
2416
2417 foo := ctx.ModuleForTests("foo", "android_common")
2418 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2419 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2420 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2421 }
2422 })
2423 }
2424}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002425
2426func TestEmbedNotice(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002427 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002428 android_app {
2429 name: "foo",
2430 srcs: ["a.java"],
2431 static_libs: ["javalib"],
2432 jni_libs: ["libjni"],
2433 notice: "APP_NOTICE",
2434 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002435 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002436 }
2437
2438 // No embed_notice flag
2439 android_app {
2440 name: "bar",
2441 srcs: ["a.java"],
2442 jni_libs: ["libjni"],
2443 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002444 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002445 }
2446
2447 // No NOTICE files
2448 android_app {
2449 name: "baz",
2450 srcs: ["a.java"],
2451 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002452 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002453 }
2454
2455 cc_library {
2456 name: "libjni",
2457 system_shared_libs: [],
2458 stl: "none",
2459 notice: "LIB_NOTICE",
Colin Cross1c93c292020-02-15 10:38:00 -08002460 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002461 }
2462
2463 java_library {
2464 name: "javalib",
2465 srcs: [
2466 ":gen",
2467 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002468 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002469 }
2470
2471 genrule {
2472 name: "gen",
2473 tools: ["gentool"],
2474 out: ["gen.java"],
2475 notice: "GENRULE_NOTICE",
2476 }
2477
2478 java_binary_host {
2479 name: "gentool",
2480 srcs: ["b.java"],
2481 notice: "TOOL_NOTICE",
2482 }
2483 `)
2484
2485 // foo has NOTICE files to process, and embed_notices is true.
2486 foo := ctx.ModuleForTests("foo", "android_common")
2487 // verify merge notices rule.
2488 mergeNotices := foo.Rule("mergeNoticesRule")
2489 noticeInputs := mergeNotices.Inputs.Strings()
2490 // TOOL_NOTICE should be excluded as it's a host module.
2491 if len(mergeNotices.Inputs) != 3 {
2492 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2493 }
2494 if !inList("APP_NOTICE", noticeInputs) {
2495 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2496 }
2497 if !inList("LIB_NOTICE", noticeInputs) {
2498 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2499 }
2500 if !inList("GENRULE_NOTICE", noticeInputs) {
2501 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2502 }
2503 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2504 res := foo.Output("package-res.apk")
2505 aapt2Flags := res.Args["flags"]
2506 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2507 if !strings.Contains(aapt2Flags, e) {
2508 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2509 }
2510
2511 // bar has NOTICE files to process, but embed_notices is not set.
2512 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002513 res = bar.Output("package-res.apk")
2514 aapt2Flags = res.Args["flags"]
2515 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2516 if strings.Contains(aapt2Flags, e) {
2517 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002518 }
2519
2520 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2521 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002522 res = baz.Output("package-res.apk")
2523 aapt2Flags = res.Args["flags"]
2524 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2525 if strings.Contains(aapt2Flags, e) {
2526 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002527 }
2528}
Colin Cross53a87f52019-06-25 13:35:30 -07002529
2530func TestUncompressDex(t *testing.T) {
2531 testCases := []struct {
2532 name string
2533 bp string
2534
2535 uncompressedPlatform bool
2536 uncompressedUnbundled bool
2537 }{
2538 {
2539 name: "normal",
2540 bp: `
2541 android_app {
2542 name: "foo",
2543 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002544 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002545 }
2546 `,
2547 uncompressedPlatform: true,
2548 uncompressedUnbundled: false,
2549 },
2550 {
2551 name: "use_embedded_dex",
2552 bp: `
2553 android_app {
2554 name: "foo",
2555 use_embedded_dex: true,
2556 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002557 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002558 }
2559 `,
2560 uncompressedPlatform: true,
2561 uncompressedUnbundled: true,
2562 },
2563 {
2564 name: "privileged",
2565 bp: `
2566 android_app {
2567 name: "foo",
2568 privileged: true,
2569 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002570 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002571 }
2572 `,
2573 uncompressedPlatform: true,
2574 uncompressedUnbundled: true,
2575 },
2576 }
2577
2578 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2579 t.Helper()
2580
Colin Cross98be1bb2019-12-13 20:41:13 -08002581 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002582 if unbundled {
2583 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2584 }
2585
Colin Cross98be1bb2019-12-13 20:41:13 -08002586 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002587
2588 run(t, ctx, config)
2589
2590 foo := ctx.ModuleForTests("foo", "android_common")
2591 dex := foo.Rule("r8")
2592 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2593 aligned := foo.MaybeRule("zipalign").Rule != nil
2594
2595 if uncompressedInDexJar != want {
2596 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2597 }
2598
2599 if aligned != want {
2600 t.Errorf("want aligned %v, got %v", want, aligned)
2601 }
2602 }
2603
2604 for _, tt := range testCases {
2605 t.Run(tt.name, func(t *testing.T) {
2606 t.Run("platform", func(t *testing.T) {
2607 test(t, tt.bp, tt.uncompressedPlatform, false)
2608 })
2609 t.Run("unbundled", func(t *testing.T) {
2610 test(t, tt.bp, tt.uncompressedUnbundled, true)
2611 })
2612 })
2613 }
2614}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002615
2616func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2617 if expectedValue != "" {
2618 expectedFlag := "--" + flagName + " " + expectedValue
2619 if !strings.Contains(aapt2Flags, expectedFlag) {
2620 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2621 }
2622 } else {
2623 unexpectedFlag := "--" + flagName
2624 if strings.Contains(aapt2Flags, unexpectedFlag) {
2625 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2626 }
2627 }
2628}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002629
2630func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungca095d72020-04-09 16:15:30 -07002631 fs := map[string][]byte{
2632 "baz/res/res/values/strings.xml": nil,
2633 "bar/res/res/values/strings.xml": nil,
2634 }
2635 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002636 runtime_resource_overlay {
2637 name: "foo",
2638 certificate: "platform",
2639 product_specific: true,
Jaewoong Jungca095d72020-04-09 16:15:30 -07002640 static_libs: ["bar"],
2641 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002642 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002643 }
2644
2645 runtime_resource_overlay {
2646 name: "foo_themed",
2647 certificate: "platform",
2648 product_specific: true,
2649 theme: "faza",
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002650 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002651 }
Jaewoong Jungca095d72020-04-09 16:15:30 -07002652
2653 android_library {
2654 name: "bar",
2655 resource_dirs: ["bar/res"],
2656 }
2657
2658 android_app {
2659 name: "baz",
2660 sdk_version: "current",
2661 resource_dirs: ["baz/res"],
2662 }
2663 `
2664 config := testAppConfig(nil, bp, fs)
2665 ctx := testContext()
2666 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002667
2668 m := ctx.ModuleForTests("foo", "android_common")
2669
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002670 // Check AAPT2 link flags.
2671 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2672 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2673 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2674 if len(absentFlags) > 0 {
2675 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2676 }
2677
Jaewoong Jungca095d72020-04-09 16:15:30 -07002678 // Check overlay.list output for static_libs dependency.
2679 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
2680 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
2681 if !inList(staticLibPackage, overlayList) {
2682 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
2683 }
2684
2685 // Check AAPT2 link flags for resource_libs dependency.
2686 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
2687 if !strings.Contains(aapt2Flags, resourceLibFlag) {
2688 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
2689 }
2690
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002691 // Check cert signing flag.
2692 signedApk := m.Output("signed/foo.apk")
2693 signingFlag := signedApk.Args["certificates"]
2694 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2695 if expected != signingFlag {
2696 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2697 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002698 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2699 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002700 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
2701 if !reflect.DeepEqual(path, expectedPath) {
2702 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
2703 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002704
2705 // Check device location.
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002706 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002707 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002708 if !reflect.DeepEqual(path, expectedPath) {
2709 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2710 }
2711
2712 // A themed module has a different device location
2713 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002714 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2715 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002716 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
2717 if !reflect.DeepEqual(path, expectedPath) {
2718 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2719 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002720
2721 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
2722 expectedOverrides := []string{"foo"}
2723 if !reflect.DeepEqual(overrides, expectedOverrides) {
2724 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
2725 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002726}
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07002727
2728func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
2729 ctx, config := testJava(t, `
2730 java_defaults {
2731 name: "rro_defaults",
2732 theme: "default_theme",
2733 product_specific: true,
2734 aaptflags: ["--keep-raw-values"],
2735 }
2736
2737 runtime_resource_overlay {
2738 name: "foo_with_defaults",
2739 defaults: ["rro_defaults"],
2740 }
2741
2742 runtime_resource_overlay {
2743 name: "foo_barebones",
2744 }
2745 `)
2746
2747 //
2748 // RRO module with defaults
2749 //
2750 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
2751
2752 // Check AAPT2 link flags.
2753 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
2754 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2755 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
2756 if len(absentFlags) > 0 {
2757 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2758 }
2759
2760 // Check device location.
2761 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2762 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
2763 if !reflect.DeepEqual(path, expectedPath) {
2764 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
2765 }
2766
2767 //
2768 // RRO module without defaults
2769 //
2770 m = ctx.ModuleForTests("foo_barebones", "android_common")
2771
2772 // Check AAPT2 link flags.
2773 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
2774 unexpectedFlags := "--keep-raw-values"
2775 if inList(unexpectedFlags, aapt2Flags) {
2776 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
2777 }
2778
2779 // Check device location.
2780 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2781 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
2782 if !reflect.DeepEqual(path, expectedPath) {
2783 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2784 }
2785}
Roshan Piusb8307962020-04-27 09:42:27 -07002786
2787func TestOverrideRuntimeResourceOverlay(t *testing.T) {
2788 ctx, _ := testJava(t, `
2789 runtime_resource_overlay {
2790 name: "foo_overlay",
2791 certificate: "platform",
2792 product_specific: true,
2793 sdk_version: "current",
2794 }
2795
2796 override_runtime_resource_overlay {
2797 name: "bar_overlay",
2798 base: "foo_overlay",
2799 package_name: "com.android.bar.overlay",
2800 target_package_name: "com.android.bar",
2801 }
2802 `)
2803
2804 expectedVariants := []struct {
2805 moduleName string
2806 variantName string
2807 apkPath string
2808 overrides []string
2809 targetVariant string
2810 packageFlag string
2811 targetPackageFlag string
2812 }{
2813 {
2814 variantName: "android_common",
2815 apkPath: "/target/product/test_device/product/overlay/foo_overlay.apk",
2816 overrides: nil,
2817 targetVariant: "android_common",
2818 packageFlag: "",
2819 targetPackageFlag: "",
2820 },
2821 {
2822 variantName: "android_common_bar_overlay",
2823 apkPath: "/target/product/test_device/product/overlay/bar_overlay.apk",
2824 overrides: []string{"foo_overlay"},
2825 targetVariant: "android_common_bar",
2826 packageFlag: "com.android.bar.overlay",
2827 targetPackageFlag: "com.android.bar",
2828 },
2829 }
2830 for _, expected := range expectedVariants {
2831 variant := ctx.ModuleForTests("foo_overlay", expected.variantName)
2832
2833 // Check the final apk name
2834 outputs := variant.AllOutputs()
2835 expectedApkPath := buildDir + expected.apkPath
2836 found := false
2837 for _, o := range outputs {
2838 if o == expectedApkPath {
2839 found = true
2840 break
2841 }
2842 }
2843 if !found {
2844 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
2845 }
2846
2847 // Check if the overrides field values are correctly aggregated.
2848 mod := variant.Module().(*RuntimeResourceOverlay)
2849 if !reflect.DeepEqual(expected.overrides, mod.properties.Overrides) {
2850 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
2851 expected.overrides, mod.properties.Overrides)
2852 }
2853
2854 // Check aapt2 flags.
2855 res := variant.Output("package-res.apk")
2856 aapt2Flags := res.Args["flags"]
2857 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
2858 checkAapt2LinkFlag(t, aapt2Flags, "rename-overlay-target-package", expected.targetPackageFlag)
2859 }
2860}