blob: 388014f4601883d2150fe0d30be15d95d1dd5b87 [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",
279 updatable: true,
280 }`,
281 },
282 {
283 name: "Stable system SDK",
284 bp: `android_app {
285 name: "foo",
286 srcs: ["a.java"],
287 sdk_version: "system_29",
288 updatable: true,
289 }`,
290 },
291 {
292 name: "Current public SDK",
293 bp: `android_app {
294 name: "foo",
295 srcs: ["a.java"],
296 sdk_version: "current",
297 updatable: true,
298 }`,
299 },
300 {
301 name: "Current system SDK",
302 bp: `android_app {
303 name: "foo",
304 srcs: ["a.java"],
305 sdk_version: "system_current",
306 updatable: true,
307 }`,
308 },
309 {
310 name: "Current module SDK",
311 bp: `android_app {
312 name: "foo",
313 srcs: ["a.java"],
314 sdk_version: "module_current",
315 updatable: true,
316 }`,
317 },
318 {
319 name: "Current core SDK",
320 bp: `android_app {
321 name: "foo",
322 srcs: ["a.java"],
323 sdk_version: "core_current",
324 updatable: true,
325 }`,
326 },
327 {
328 name: "No Platform APIs",
329 bp: `android_app {
330 name: "foo",
331 srcs: ["a.java"],
332 platform_apis: true,
333 updatable: true,
334 }`,
335 expectedError: "Updatable apps must use stable SDKs",
336 },
337 {
338 name: "No Core Platform APIs",
339 bp: `android_app {
340 name: "foo",
341 srcs: ["a.java"],
342 sdk_version: "core_platform",
343 updatable: true,
344 }`,
345 expectedError: "Updatable apps must use stable SDKs",
346 },
347 {
348 name: "No unspecified APIs",
349 bp: `android_app {
350 name: "foo",
351 srcs: ["a.java"],
352 updatable: true,
353 }`,
354 expectedError: "Updatable apps must use stable SDK",
355 },
356 }
357
358 for _, test := range testCases {
359 t.Run(test.name, func(t *testing.T) {
360 if test.expectedError == "" {
361 testJava(t, test.bp)
362 } else {
363 testJavaError(t, test.expectedError, test.bp)
364 }
365 })
366 }
367}
368
Colin Cross0ddae7f2019-02-07 15:30:01 -0800369func TestResourceDirs(t *testing.T) {
370 testCases := []struct {
371 name string
372 prop string
373 resources []string
374 }{
375 {
376 name: "no resource_dirs",
377 prop: "",
378 resources: []string{"res/res/values/strings.xml"},
379 },
380 {
381 name: "resource_dirs",
382 prop: `resource_dirs: ["res"]`,
383 resources: []string{"res/res/values/strings.xml"},
384 },
385 {
386 name: "empty resource_dirs",
387 prop: `resource_dirs: []`,
388 resources: nil,
389 },
390 }
391
392 fs := map[string][]byte{
393 "res/res/values/strings.xml": nil,
394 }
395
396 bp := `
397 android_app {
398 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900399 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800400 %s
401 }
402 `
403
404 for _, testCase := range testCases {
405 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800406 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
407 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800408 run(t, ctx, config)
409
410 module := ctx.ModuleForTests("foo", "android_common")
411 resourceList := module.MaybeOutput("aapt2/res.list")
412
413 var resources []string
414 if resourceList.Rule != nil {
415 for _, compiledResource := range resourceList.Inputs.Strings() {
416 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
417 }
418 }
419
420 if !reflect.DeepEqual(resources, testCase.resources) {
421 t.Errorf("expected resource files %q, got %q",
422 testCase.resources, resources)
423 }
424 })
425 }
426}
427
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800428func TestLibraryAssets(t *testing.T) {
429 bp := `
430 android_app {
431 name: "foo",
432 sdk_version: "current",
433 static_libs: ["lib1", "lib2", "lib3"],
434 }
435
436 android_library {
437 name: "lib1",
438 sdk_version: "current",
439 asset_dirs: ["assets_a"],
440 }
441
442 android_library {
443 name: "lib2",
444 sdk_version: "current",
445 }
446
447 android_library {
448 name: "lib3",
449 sdk_version: "current",
450 static_libs: ["lib4"],
451 }
452
453 android_library {
454 name: "lib4",
455 sdk_version: "current",
456 asset_dirs: ["assets_b"],
457 }
458 `
459
460 testCases := []struct {
461 name string
462 assetFlag string
463 assetPackages []string
464 }{
465 {
466 name: "foo",
467 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
468 assetPackages: []string{
469 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
470 buildDir + "/.intermediates/lib1/android_common/assets.zip",
471 buildDir + "/.intermediates/lib3/android_common/assets.zip",
472 },
473 },
474 {
475 name: "lib1",
476 assetFlag: "-A assets_a",
477 },
478 {
479 name: "lib2",
480 },
481 {
482 name: "lib3",
483 assetPackages: []string{
484 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
485 buildDir + "/.intermediates/lib4/android_common/assets.zip",
486 },
487 },
488 {
489 name: "lib4",
490 assetFlag: "-A assets_b",
491 },
492 }
493 ctx := testApp(t, bp)
494
495 for _, test := range testCases {
496 t.Run(test.name, func(t *testing.T) {
497 m := ctx.ModuleForTests(test.name, "android_common")
498
499 // Check asset flag in aapt2 link flags
500 var aapt2link android.TestingBuildParams
501 if len(test.assetPackages) > 0 {
502 aapt2link = m.Output("aapt2/package-res.apk")
503 } else {
504 aapt2link = m.Output("package-res.apk")
505 }
506 aapt2Flags := aapt2link.Args["flags"]
507 if test.assetFlag != "" {
508 if !strings.Contains(aapt2Flags, test.assetFlag) {
509 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
510 }
511 } else {
512 if strings.Contains(aapt2Flags, " -A ") {
513 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
514 }
515 }
516
517 // Check asset merge rule.
518 if len(test.assetPackages) > 0 {
519 mergeAssets := m.Output("package-res.apk")
520 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
521 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
522 mergeAssets.Inputs.Strings(), test.assetPackages)
523 }
524 }
525 })
526 }
527}
528
Colin Crossbec85302019-02-13 13:15:46 -0800529func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800530 testCases := []struct {
531 name string
532 enforceRROTargets []string
533 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800534 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800535 overlayFiles map[string][]string
536 rroDirs map[string][]string
537 }{
538 {
539 name: "no RRO",
540 enforceRROTargets: nil,
541 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800542 resourceFiles: map[string][]string{
543 "foo": nil,
544 "bar": {"bar/res/res/values/strings.xml"},
545 "lib": nil,
546 "lib2": {"lib2/res/res/values/strings.xml"},
547 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800548 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800549 "foo": {
550 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800551 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000552 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800553 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800554 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
555 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000556 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800557 },
Colin Crossbec85302019-02-13 13:15:46 -0800558 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800559 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
560 "device/vendor/blah/overlay/bar/res/values/strings.xml",
561 },
Colin Crossbec85302019-02-13 13:15:46 -0800562 "lib": {
563 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
564 "lib/res/res/values/strings.xml",
565 "device/vendor/blah/overlay/lib/res/values/strings.xml",
566 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800567 },
568 rroDirs: map[string][]string{
569 "foo": nil,
570 "bar": nil,
571 },
572 },
573 {
574 name: "enforce RRO on foo",
575 enforceRROTargets: []string{"foo"},
576 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800577 resourceFiles: map[string][]string{
578 "foo": nil,
579 "bar": {"bar/res/res/values/strings.xml"},
580 "lib": nil,
581 "lib2": {"lib2/res/res/values/strings.xml"},
582 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800583 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800584 "foo": {
585 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800586 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000587 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800588 "foo/res/res/values/strings.xml",
589 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
590 },
Colin Crossbec85302019-02-13 13:15:46 -0800591 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800592 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
593 "device/vendor/blah/overlay/bar/res/values/strings.xml",
594 },
Colin Crossbec85302019-02-13 13:15:46 -0800595 "lib": {
596 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
597 "lib/res/res/values/strings.xml",
598 "device/vendor/blah/overlay/lib/res/values/strings.xml",
599 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800600 },
Colin Crossc1c37552019-01-31 11:42:41 -0800601
Colin Cross5c4791c2019-02-01 11:44:44 -0800602 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800603 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000604 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800605 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
606 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000607 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800608 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800609 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800610 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800611 },
612 },
613 {
614 name: "enforce RRO on all",
615 enforceRROTargets: []string{"*"},
616 enforceRROExcludedOverlays: []string{
617 // Excluding specific apps/res directories also allowed.
618 "device/vendor/blah/static_overlay/foo",
619 "device/vendor/blah/static_overlay/bar/res",
620 },
Colin Crossbec85302019-02-13 13:15:46 -0800621 resourceFiles: map[string][]string{
622 "foo": nil,
623 "bar": {"bar/res/res/values/strings.xml"},
624 "lib": nil,
625 "lib2": {"lib2/res/res/values/strings.xml"},
626 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800627 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800628 "foo": {
629 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800630 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000631 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800632 "foo/res/res/values/strings.xml",
633 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
634 },
Colin Crossbec85302019-02-13 13:15:46 -0800635 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
636 "lib": {
637 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
638 "lib/res/res/values/strings.xml",
639 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800640 },
641 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800642 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000643 "device:device/vendor/blah/overlay/foo/res",
644 "product:product/vendor/blah/overlay/foo/res",
645 // Lib dep comes after the direct deps
646 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800647 },
Anton Hansson53c88442019-03-18 15:53:16 +0000648 "bar": {"device:device/vendor/blah/overlay/bar/res"},
649 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800650 },
651 },
652 }
653
Anton Hansson53c88442019-03-18 15:53:16 +0000654 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800655 "device/vendor/blah/overlay",
656 "device/vendor/blah/overlay2",
657 "device/vendor/blah/static_overlay",
658 }
659
Anton Hansson53c88442019-03-18 15:53:16 +0000660 productResourceOverlays := []string{
661 "product/vendor/blah/overlay",
662 }
663
Colin Cross890ff552017-11-30 20:13:19 -0800664 fs := map[string][]byte{
665 "foo/res/res/values/strings.xml": nil,
666 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800667 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800668 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800669 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
670 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800671 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800672 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
673 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
674 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000675 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800676 }
677
678 bp := `
679 android_app {
680 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900681 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800682 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000683 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800684 }
685
686 android_app {
687 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900688 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800689 resource_dirs: ["bar/res"],
690 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800691
692 android_library {
693 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900694 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800695 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800696 static_libs: ["lib2"],
697 }
698
699 android_library {
700 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900701 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800702 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800703 }
Anton Hansson53c88442019-03-18 15:53:16 +0000704
705 // This library has the same resources as lib (should not lead to dupe RROs)
706 android_library {
707 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900708 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000709 resource_dirs: ["lib/res"]
710 }
Colin Cross890ff552017-11-30 20:13:19 -0800711 `
712
Colin Cross5c4791c2019-02-01 11:44:44 -0800713 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800714 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800715 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000716 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
717 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800718 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800719 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800720 }
721 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800722 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800723 }
724
Colin Cross98be1bb2019-12-13 20:41:13 -0800725 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800726 run(t, ctx, config)
727
Colin Crossbec85302019-02-13 13:15:46 -0800728 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
729 for _, o := range list {
730 res := module.MaybeOutput(o)
731 if res.Rule != nil {
732 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
733 // verify the inputs to the .arsc.flat rule.
734 files = append(files, res.Inputs.Strings()...)
735 } else {
736 // Otherwise, verify the full path to the output of the other module
737 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000738 }
Colin Cross890ff552017-11-30 20:13:19 -0800739 }
Colin Crossbec85302019-02-13 13:15:46 -0800740 return files
Colin Cross890ff552017-11-30 20:13:19 -0800741 }
742
Colin Crossbec85302019-02-13 13:15:46 -0800743 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
744 module := ctx.ModuleForTests(moduleName, "android_common")
745 resourceList := module.MaybeOutput("aapt2/res.list")
746 if resourceList.Rule != nil {
747 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000748 }
Colin Crossbec85302019-02-13 13:15:46 -0800749 overlayList := module.MaybeOutput("aapt2/overlay.list")
750 if overlayList.Rule != nil {
751 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
752 }
753
Anton Hansson53c88442019-03-18 15:53:16 +0000754 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
755 var prefix string
756 if d.overlayType == device {
757 prefix = "device:"
758 } else if d.overlayType == product {
759 prefix = "product:"
760 } else {
761 t.Fatalf("Unexpected overlayType %d", d.overlayType)
762 }
763 rroDirs = append(rroDirs, prefix+d.path.String())
764 }
Colin Crossbec85302019-02-13 13:15:46 -0800765
766 return resourceFiles, overlayFiles, rroDirs
767 }
768
769 modules := []string{"foo", "bar", "lib", "lib2"}
770 for _, module := range modules {
771 resourceFiles, overlayFiles, rroDirs := getResources(module)
772
773 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
774 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
775 module, testCase.resourceFiles[module], resourceFiles)
776 }
777 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
778 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
779 module, testCase.overlayFiles[module], overlayFiles)
780 }
781 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +0000782 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -0800783 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +0000784 }
Colin Cross890ff552017-11-30 20:13:19 -0800785 }
Colin Cross890ff552017-11-30 20:13:19 -0800786 })
787 }
788}
Colin Crossd09b0b62018-04-18 11:06:47 -0700789
790func TestAppSdkVersion(t *testing.T) {
791 testCases := []struct {
792 name string
793 sdkVersion string
794 platformSdkInt int
795 platformSdkCodename string
796 platformSdkFinal bool
797 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +0900798 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -0700799 }{
800 {
801 name: "current final SDK",
802 sdkVersion: "current",
803 platformSdkInt: 27,
804 platformSdkCodename: "REL",
805 platformSdkFinal: true,
806 expectedMinSdkVersion: "27",
807 },
808 {
809 name: "current non-final SDK",
810 sdkVersion: "current",
811 platformSdkInt: 27,
812 platformSdkCodename: "OMR1",
813 platformSdkFinal: false,
814 expectedMinSdkVersion: "OMR1",
815 },
816 {
817 name: "default final SDK",
818 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900819 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700820 platformSdkInt: 27,
821 platformSdkCodename: "REL",
822 platformSdkFinal: true,
823 expectedMinSdkVersion: "27",
824 },
825 {
826 name: "default non-final SDK",
827 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900828 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700829 platformSdkInt: 27,
830 platformSdkCodename: "OMR1",
831 platformSdkFinal: false,
832 expectedMinSdkVersion: "OMR1",
833 },
834 {
835 name: "14",
836 sdkVersion: "14",
837 expectedMinSdkVersion: "14",
838 },
839 }
840
841 for _, moduleType := range []string{"android_app", "android_library"} {
842 for _, test := range testCases {
843 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +0900844 platformApiProp := ""
845 if test.platformApis {
846 platformApiProp = "platform_apis: true,"
847 }
Colin Crossd09b0b62018-04-18 11:06:47 -0700848 bp := fmt.Sprintf(`%s {
849 name: "foo",
850 srcs: ["a.java"],
851 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900852 %s
853 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -0700854
Colin Cross98be1bb2019-12-13 20:41:13 -0800855 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -0700856 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
857 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
858 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
859
Colin Cross98be1bb2019-12-13 20:41:13 -0800860 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -0700861
862 run(t, ctx, config)
863
864 foo := ctx.ModuleForTests("foo", "android_common")
865 link := foo.Output("package-res.apk")
866 linkFlags := strings.Split(link.Args["flags"], " ")
867 min := android.IndexList("--min-sdk-version", linkFlags)
868 target := android.IndexList("--target-sdk-version", linkFlags)
869
870 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
871 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
872 }
873
874 gotMinSdkVersion := linkFlags[min+1]
875 gotTargetSdkVersion := linkFlags[target+1]
876
877 if gotMinSdkVersion != test.expectedMinSdkVersion {
878 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
879 test.expectedMinSdkVersion, gotMinSdkVersion)
880 }
881
882 if gotTargetSdkVersion != test.expectedMinSdkVersion {
883 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
884 test.expectedMinSdkVersion, gotTargetSdkVersion)
885 }
886 })
887 }
888 }
889}
Colin Crossa4f08812018-10-02 22:03:40 -0700890
Paul Duffin50c217c2019-06-12 13:25:22 +0100891func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700892 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +0100893 cc_library {
894 name: "libjni",
895 system_shared_libs: [],
896 stl: "none",
897 }
898
899 android_test {
900 name: "test",
901 sdk_version: "core_platform",
902 jni_libs: ["libjni"],
903 }
904
905 android_test {
906 name: "test_first",
907 sdk_version: "core_platform",
908 compile_multilib: "first",
909 jni_libs: ["libjni"],
910 }
911
912 android_test {
913 name: "test_both",
914 sdk_version: "core_platform",
915 compile_multilib: "both",
916 jni_libs: ["libjni"],
917 }
918
919 android_test {
920 name: "test_32",
921 sdk_version: "core_platform",
922 compile_multilib: "32",
923 jni_libs: ["libjni"],
924 }
925
926 android_test {
927 name: "test_64",
928 sdk_version: "core_platform",
929 compile_multilib: "64",
930 jni_libs: ["libjni"],
931 }
932 `)
933
934 testCases := []struct {
935 name string
936 abis []string
937 }{
938 {"test", []string{"arm64-v8a"}},
939 {"test_first", []string{"arm64-v8a"}},
940 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
941 {"test_32", []string{"armeabi-v7a"}},
942 {"test_64", []string{"arm64-v8a"}},
943 }
944
945 for _, test := range testCases {
946 t.Run(test.name, func(t *testing.T) {
947 app := ctx.ModuleForTests(test.name, "android_common")
948 jniLibZip := app.Output("jnilibs.zip")
949 var abis []string
950 args := strings.Fields(jniLibZip.Args["jarArgs"])
951 for i := 0; i < len(args); i++ {
952 if args[i] == "-P" {
953 abis = append(abis, filepath.Base(args[i+1]))
954 i++
955 }
956 }
957 if !reflect.DeepEqual(abis, test.abis) {
958 t.Errorf("want abis %v, got %v", test.abis, abis)
959 }
960 })
961 }
962}
963
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900964func TestAppSdkVersionByPartition(t *testing.T) {
965 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
966 android_app {
967 name: "foo",
968 srcs: ["a.java"],
969 vendor: true,
970 platform_apis: true,
971 }
972 `)
973
974 testJava(t, `
975 android_app {
976 name: "bar",
977 srcs: ["b.java"],
978 platform_apis: true,
979 }
980 `)
981
982 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900983 bp := `
984 android_app {
985 name: "foo",
986 srcs: ["a.java"],
987 product_specific: true,
988 platform_apis: true,
989 }
990 `
Colin Cross98be1bb2019-12-13 20:41:13 -0800991
992 config := testAppConfig(nil, bp, nil)
993 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900994 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -0800995 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 +0900996 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -0800997 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900998 }
999 }
1000}
1001
Paul Duffin50c217c2019-06-12 13:25:22 +01001002func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001003 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +01001004 cc_library {
1005 name: "libjni",
1006 system_shared_libs: [],
1007 stl: "none",
Colin Cross1c93c292020-02-15 10:38:00 -08001008 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001009 }
1010
1011 android_app {
1012 name: "app",
1013 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001014 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001015 }
1016
1017 android_app {
1018 name: "app_noembed",
1019 jni_libs: ["libjni"],
1020 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001021 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001022 }
1023
1024 android_app {
1025 name: "app_embed",
1026 jni_libs: ["libjni"],
1027 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001028 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +01001029 }
1030
1031 android_test {
1032 name: "test",
1033 sdk_version: "core_platform",
1034 jni_libs: ["libjni"],
1035 }
1036
1037 android_test {
1038 name: "test_noembed",
1039 sdk_version: "core_platform",
1040 jni_libs: ["libjni"],
1041 use_embedded_native_libs: false,
1042 }
1043
1044 android_test_helper_app {
1045 name: "test_helper",
1046 sdk_version: "core_platform",
1047 jni_libs: ["libjni"],
1048 }
1049
1050 android_test_helper_app {
1051 name: "test_helper_noembed",
1052 sdk_version: "core_platform",
1053 jni_libs: ["libjni"],
1054 use_embedded_native_libs: false,
1055 }
1056 `)
1057
1058 testCases := []struct {
1059 name string
1060 packaged bool
1061 compressed bool
1062 }{
1063 {"app", false, false},
1064 {"app_noembed", false, false},
1065 {"app_embed", true, false},
1066 {"test", true, false},
1067 {"test_noembed", true, true},
1068 {"test_helper", true, false},
1069 {"test_helper_noembed", true, true},
1070 }
1071
1072 for _, test := range testCases {
1073 t.Run(test.name, func(t *testing.T) {
1074 app := ctx.ModuleForTests(test.name, "android_common")
1075 jniLibZip := app.MaybeOutput("jnilibs.zip")
1076 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
1077 t.Errorf("expected jni packaged %v, got %v", w, g)
1078 }
1079
1080 if jniLibZip.Rule != nil {
1081 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
1082 t.Errorf("expected jni compressed %v, got %v", w, g)
1083 }
1084 }
1085 })
1086 }
Colin Cross47fa9d32019-03-26 10:51:39 -07001087}
1088
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001089func TestCertificates(t *testing.T) {
1090 testCases := []struct {
1091 name string
1092 bp string
1093 certificateOverride string
1094 expected string
1095 }{
1096 {
1097 name: "default",
1098 bp: `
1099 android_app {
1100 name: "foo",
1101 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001102 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001103 }
1104 `,
1105 certificateOverride: "",
Dan Willemsen412160e2019-04-09 21:36:26 -07001106 expected: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001107 },
1108 {
1109 name: "module certificate property",
1110 bp: `
1111 android_app {
1112 name: "foo",
1113 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001114 certificate: ":new_certificate",
1115 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001116 }
1117
1118 android_app_certificate {
1119 name: "new_certificate",
1120 certificate: "cert/new_cert",
1121 }
1122 `,
1123 certificateOverride: "",
1124 expected: "cert/new_cert.x509.pem cert/new_cert.pk8",
1125 },
1126 {
1127 name: "path certificate property",
1128 bp: `
1129 android_app {
1130 name: "foo",
1131 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001132 certificate: "expiredkey",
1133 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001134 }
1135 `,
1136 certificateOverride: "",
Dan Willemsen412160e2019-04-09 21:36:26 -07001137 expected: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001138 },
1139 {
1140 name: "certificate overrides",
1141 bp: `
1142 android_app {
1143 name: "foo",
1144 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001145 certificate: "expiredkey",
1146 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001147 }
1148
1149 android_app_certificate {
1150 name: "new_certificate",
1151 certificate: "cert/new_cert",
1152 }
1153 `,
1154 certificateOverride: "foo:new_certificate",
1155 expected: "cert/new_cert.x509.pem cert/new_cert.pk8",
1156 },
1157 }
1158
1159 for _, test := range testCases {
1160 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001161 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001162 if test.certificateOverride != "" {
1163 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1164 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001165 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001166
1167 run(t, ctx, config)
1168 foo := ctx.ModuleForTests("foo", "android_common")
1169
1170 signapk := foo.Output("foo.apk")
1171 signFlags := signapk.Args["certificates"]
1172 if test.expected != signFlags {
1173 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1174 }
1175 })
1176 }
1177}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001178
Songchun Fan688de9a2020-03-24 20:32:24 -07001179func TestRequestV4SigningFlag(t *testing.T) {
1180 testCases := []struct {
1181 name string
1182 bp string
1183 expected string
1184 }{
1185 {
1186 name: "default",
1187 bp: `
1188 android_app {
1189 name: "foo",
1190 srcs: ["a.java"],
1191 sdk_version: "current",
1192 }
1193 `,
1194 expected: "",
1195 },
1196 {
1197 name: "default",
1198 bp: `
1199 android_app {
1200 name: "foo",
1201 srcs: ["a.java"],
1202 sdk_version: "current",
1203 v4_signature: false,
1204 }
1205 `,
1206 expected: "",
1207 },
1208 {
1209 name: "module certificate property",
1210 bp: `
1211 android_app {
1212 name: "foo",
1213 srcs: ["a.java"],
1214 sdk_version: "current",
1215 v4_signature: true,
1216 }
1217 `,
1218 expected: "--enable-v4",
1219 },
1220 }
1221
1222 for _, test := range testCases {
1223 t.Run(test.name, func(t *testing.T) {
1224 config := testAppConfig(nil, test.bp, nil)
1225 ctx := testContext()
1226
1227 run(t, ctx, config)
1228 foo := ctx.ModuleForTests("foo", "android_common")
1229
1230 signapk := foo.Output("foo.apk")
1231 signFlags := signapk.Args["flags"]
1232 if test.expected != signFlags {
1233 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1234 }
1235 })
1236 }
1237}
1238
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001239func TestPackageNameOverride(t *testing.T) {
1240 testCases := []struct {
1241 name string
1242 bp string
1243 packageNameOverride string
1244 expected []string
1245 }{
1246 {
1247 name: "default",
1248 bp: `
1249 android_app {
1250 name: "foo",
1251 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001252 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001253 }
1254 `,
1255 packageNameOverride: "",
1256 expected: []string{
1257 buildDir + "/.intermediates/foo/android_common/foo.apk",
1258 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1259 },
1260 },
1261 {
1262 name: "overridden",
1263 bp: `
1264 android_app {
1265 name: "foo",
1266 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001267 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001268 }
1269 `,
1270 packageNameOverride: "foo:bar",
1271 expected: []string{
1272 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001273 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001274 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1275 },
1276 },
1277 }
1278
1279 for _, test := range testCases {
1280 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001281 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001282 if test.packageNameOverride != "" {
1283 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1284 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001285 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001286
1287 run(t, ctx, config)
1288 foo := ctx.ModuleForTests("foo", "android_common")
1289
1290 outputs := foo.AllOutputs()
1291 outputMap := make(map[string]bool)
1292 for _, o := range outputs {
1293 outputMap[o] = true
1294 }
1295 for _, e := range test.expected {
1296 if _, exist := outputMap[e]; !exist {
1297 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1298 }
1299 }
1300 })
1301 }
1302}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001303
1304func TestInstrumentationTargetOverridden(t *testing.T) {
1305 bp := `
1306 android_app {
1307 name: "foo",
1308 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001309 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001310 }
1311
1312 android_test {
1313 name: "bar",
1314 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001315 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001316 }
1317 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001318 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001319 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001320 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001321
1322 run(t, ctx, config)
1323
1324 bar := ctx.ModuleForTests("bar", "android_common")
1325 res := bar.Output("package-res.apk")
1326 aapt2Flags := res.Args["flags"]
1327 e := "--rename-instrumentation-target-package org.dandroid.bp"
1328 if !strings.Contains(aapt2Flags, e) {
1329 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1330 }
1331}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001332
1333func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001334 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001335 android_app {
1336 name: "foo",
1337 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001338 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001339 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001340 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001341 }
1342
1343 override_android_app {
1344 name: "bar",
1345 base: "foo",
1346 certificate: ":new_certificate",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001347 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001348 }
1349
1350 android_app_certificate {
1351 name: "new_certificate",
1352 certificate: "cert/new_cert",
1353 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001354
1355 override_android_app {
1356 name: "baz",
1357 base: "foo",
1358 package_name: "org.dandroid.bp",
1359 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001360 `)
1361
1362 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001363 moduleName string
1364 variantName string
1365 apkName string
1366 apkPath string
1367 signFlag string
1368 overrides []string
1369 aaptFlag string
1370 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001371 }{
1372 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001373 moduleName: "foo",
1374 variantName: "android_common",
1375 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
1376 signFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1377 overrides: []string{"qux"},
1378 aaptFlag: "",
1379 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001380 },
1381 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001382 moduleName: "bar",
1383 variantName: "android_common_bar",
1384 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
1385 signFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1386 overrides: []string{"qux", "foo"},
1387 aaptFlag: "",
1388 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001389 },
1390 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001391 moduleName: "baz",
1392 variantName: "android_common_baz",
1393 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
1394 signFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1395 overrides: []string{"qux", "foo"},
1396 aaptFlag: "--rename-manifest-package org.dandroid.bp",
1397 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001398 },
1399 }
1400 for _, expected := range expectedVariants {
1401 variant := ctx.ModuleForTests("foo", expected.variantName)
1402
1403 // Check the final apk name
1404 outputs := variant.AllOutputs()
1405 expectedApkPath := buildDir + expected.apkPath
1406 found := false
1407 for _, o := range outputs {
1408 if o == expectedApkPath {
1409 found = true
1410 break
1411 }
1412 }
1413 if !found {
1414 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1415 }
1416
1417 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001418 signapk := variant.Output(expected.moduleName + ".apk")
Jaewoong Jung525443a2019-02-28 15:35:54 -08001419 signFlag := signapk.Args["certificates"]
1420 if expected.signFlag != signFlag {
1421 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.signFlag, signFlag)
1422 }
1423
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001424 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001425 mod := variant.Module().(*AndroidApp)
1426 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1427 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1428 expected.overrides, mod.appProperties.Overrides)
1429 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001430
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001431 // Test Overridable property: Logging_parent
1432 logging_parent := mod.aapt.LoggingParent
1433 if expected.logging_parent != logging_parent {
1434 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1435 expected.logging_parent, logging_parent)
1436 }
1437
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001438 // Check the package renaming flag, if exists.
1439 res := variant.Output("package-res.apk")
1440 aapt2Flags := res.Args["flags"]
1441 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1442 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1443 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001444 }
1445}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001446
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001447func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001448 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001449 android_app {
1450 name: "foo",
1451 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001452 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001453 }
1454
1455 override_android_app {
1456 name: "bar",
1457 base: "foo",
1458 package_name: "org.dandroid.bp",
1459 }
1460
1461 android_test {
1462 name: "baz",
1463 srcs: ["b.java"],
1464 instrumentation_for: "foo",
1465 }
1466
1467 android_test {
1468 name: "qux",
1469 srcs: ["b.java"],
1470 instrumentation_for: "bar",
1471 }
1472 `)
1473
1474 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1475 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1476 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1477 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1478 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1479 }
1480
1481 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1482 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1483 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1484 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1485 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1486 }
1487}
1488
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001489func TestOverrideAndroidTest(t *testing.T) {
1490 ctx, _ := testJava(t, `
1491 android_app {
1492 name: "foo",
1493 srcs: ["a.java"],
1494 package_name: "com.android.foo",
1495 sdk_version: "current",
1496 }
1497
1498 override_android_app {
1499 name: "bar",
1500 base: "foo",
1501 package_name: "com.android.bar",
1502 }
1503
1504 android_test {
1505 name: "foo_test",
1506 srcs: ["b.java"],
1507 instrumentation_for: "foo",
1508 }
1509
1510 override_android_test {
1511 name: "bar_test",
1512 base: "foo_test",
1513 package_name: "com.android.bar.test",
1514 instrumentation_for: "bar",
1515 instrumentation_target_package: "com.android.bar",
1516 }
1517 `)
1518
1519 expectedVariants := []struct {
1520 moduleName string
1521 variantName string
1522 apkPath string
1523 overrides []string
1524 targetVariant string
1525 packageFlag string
1526 targetPackageFlag string
1527 }{
1528 {
1529 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001530 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001531 overrides: nil,
1532 targetVariant: "android_common",
1533 packageFlag: "",
1534 targetPackageFlag: "",
1535 },
1536 {
1537 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001538 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001539 overrides: []string{"foo_test"},
1540 targetVariant: "android_common_bar",
1541 packageFlag: "com.android.bar.test",
1542 targetPackageFlag: "com.android.bar",
1543 },
1544 }
1545 for _, expected := range expectedVariants {
1546 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1547
1548 // Check the final apk name
1549 outputs := variant.AllOutputs()
1550 expectedApkPath := buildDir + expected.apkPath
1551 found := false
1552 for _, o := range outputs {
1553 if o == expectedApkPath {
1554 found = true
1555 break
1556 }
1557 }
1558 if !found {
1559 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1560 }
1561
1562 // Check if the overrides field values are correctly aggregated.
1563 mod := variant.Module().(*AndroidTest)
1564 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1565 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1566 expected.overrides, mod.appProperties.Overrides)
1567 }
1568
1569 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1570 javac := variant.Rule("javac")
1571 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1572 if !strings.Contains(javac.Args["classpath"], turbine) {
1573 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1574 }
1575
1576 // Check aapt2 flags.
1577 res := variant.Output("package-res.apk")
1578 aapt2Flags := res.Args["flags"]
1579 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1580 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1581 }
1582}
1583
Jaewoong Jung39982342020-01-14 10:27:18 -08001584func TestAndroidTest_FixTestConfig(t *testing.T) {
1585 ctx, _ := testJava(t, `
1586 android_app {
1587 name: "foo",
1588 srcs: ["a.java"],
1589 package_name: "com.android.foo",
1590 sdk_version: "current",
1591 }
1592
1593 android_test {
1594 name: "foo_test",
1595 srcs: ["b.java"],
1596 instrumentation_for: "foo",
1597 }
1598
1599 android_test {
1600 name: "bar_test",
1601 srcs: ["b.java"],
1602 package_name: "com.android.bar.test",
1603 instrumentation_for: "foo",
1604 }
1605
1606 override_android_test {
1607 name: "baz_test",
1608 base: "foo_test",
1609 package_name: "com.android.baz.test",
1610 }
1611 `)
1612
1613 testCases := []struct {
1614 moduleName string
1615 variantName string
1616 expectedFlags []string
1617 }{
1618 {
1619 moduleName: "foo_test",
1620 variantName: "android_common",
1621 },
1622 {
1623 moduleName: "bar_test",
1624 variantName: "android_common",
1625 expectedFlags: []string{
1626 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
1627 "--package-name com.android.bar.test",
1628 },
1629 },
1630 {
1631 moduleName: "foo_test",
1632 variantName: "android_common_baz_test",
1633 expectedFlags: []string{
1634 "--manifest " + buildDir +
1635 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
1636 "--package-name com.android.baz.test",
1637 "--test-file-name baz_test.apk",
1638 },
1639 },
1640 }
1641
1642 for _, test := range testCases {
1643 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
1644 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
1645
1646 if len(test.expectedFlags) > 0 {
1647 if params.Rule == nil {
1648 t.Errorf("test_config_fixer was expected to run, but didn't")
1649 } else {
1650 for _, flag := range test.expectedFlags {
1651 if !strings.Contains(params.RuleParams.Command, flag) {
1652 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
1653 }
1654 }
1655 }
1656 } else {
1657 if params.Rule != nil {
1658 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
1659 }
1660 }
1661
1662 }
1663}
1664
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001665func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001666 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001667 android_app_import {
1668 name: "foo",
1669 apk: "prebuilts/apk/app.apk",
1670 certificate: "platform",
1671 dex_preopt: {
1672 enabled: true,
1673 },
1674 }
1675 `)
1676
1677 variant := ctx.ModuleForTests("foo", "android_common")
1678
1679 // Check dexpreopt outputs.
1680 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1681 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1682 t.Errorf("can't find dexpreopt outputs")
1683 }
1684
1685 // Check cert signing flag.
1686 signedApk := variant.Output("signed/foo.apk")
1687 signingFlag := signedApk.Args["certificates"]
1688 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
1689 if expected != signingFlag {
1690 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1691 }
1692}
1693
1694func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001695 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001696 android_app_import {
1697 name: "foo",
1698 apk: "prebuilts/apk/app.apk",
1699 certificate: "platform",
1700 dex_preopt: {
1701 enabled: false,
1702 },
1703 }
1704 `)
1705
1706 variant := ctx.ModuleForTests("foo", "android_common")
1707
1708 // Check dexpreopt outputs. They shouldn't exist.
1709 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
1710 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
1711 t.Errorf("dexpreopt shouldn't have run.")
1712 }
1713}
1714
1715func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001716 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001717 android_app_import {
1718 name: "foo",
1719 apk: "prebuilts/apk/app.apk",
1720 presigned: true,
1721 dex_preopt: {
1722 enabled: true,
1723 },
1724 }
1725 `)
1726
1727 variant := ctx.ModuleForTests("foo", "android_common")
1728
1729 // Check dexpreopt outputs.
1730 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1731 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1732 t.Errorf("can't find dexpreopt outputs")
1733 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01001734 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001735 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
1736 t.Errorf("signing rule shouldn't be included.")
1737 }
1738 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
1739 t.Errorf("can't find aligning rule")
1740 }
1741}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001742
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001743func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
1744 ctx, _ := testJava(t, `
1745 android_app_import {
1746 name: "foo",
1747 apk: "prebuilts/apk/app.apk",
1748 default_dev_cert: true,
1749 dex_preopt: {
1750 enabled: true,
1751 },
1752 }
1753 `)
1754
1755 variant := ctx.ModuleForTests("foo", "android_common")
1756
1757 // Check dexpreopt outputs.
1758 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1759 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1760 t.Errorf("can't find dexpreopt outputs")
1761 }
1762
1763 // Check cert signing flag.
1764 signedApk := variant.Output("signed/foo.apk")
1765 signingFlag := signedApk.Args["certificates"]
1766 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
1767 if expected != signingFlag {
1768 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1769 }
1770}
1771
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001772func TestAndroidAppImport_DpiVariants(t *testing.T) {
1773 bp := `
1774 android_app_import {
1775 name: "foo",
1776 apk: "prebuilts/apk/app.apk",
1777 dpi_variants: {
1778 xhdpi: {
1779 apk: "prebuilts/apk/app_xhdpi.apk",
1780 },
1781 xxhdpi: {
1782 apk: "prebuilts/apk/app_xxhdpi.apk",
1783 },
1784 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001785 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001786 dex_preopt: {
1787 enabled: true,
1788 },
1789 }
1790 `
1791 testCases := []struct {
1792 name string
1793 aaptPreferredConfig *string
1794 aaptPrebuiltDPI []string
1795 expected string
1796 }{
1797 {
1798 name: "no preferred",
1799 aaptPreferredConfig: nil,
1800 aaptPrebuiltDPI: []string{},
1801 expected: "prebuilts/apk/app.apk",
1802 },
1803 {
1804 name: "AAPTPreferredConfig matches",
1805 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001806 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001807 expected: "prebuilts/apk/app_xhdpi.apk",
1808 },
1809 {
1810 name: "AAPTPrebuiltDPI matches",
1811 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1812 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
1813 expected: "prebuilts/apk/app_xxhdpi.apk",
1814 },
1815 {
1816 name: "non-first AAPTPrebuiltDPI matches",
1817 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1818 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
1819 expected: "prebuilts/apk/app_xhdpi.apk",
1820 },
1821 {
1822 name: "no matches",
1823 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1824 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
1825 expected: "prebuilts/apk/app.apk",
1826 },
1827 }
1828
1829 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
1830 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08001831 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001832 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
1833 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08001834 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001835
1836 run(t, ctx, config)
1837
1838 variant := ctx.ModuleForTests("foo", "android_common")
1839 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
1840 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
1841 if len(matches) != 2 {
1842 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
1843 }
1844 if test.expected != matches[1] {
1845 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
1846 }
1847 }
1848}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001849
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07001850func TestAndroidAppImport_Filename(t *testing.T) {
1851 ctx, config := testJava(t, `
1852 android_app_import {
1853 name: "foo",
1854 apk: "prebuilts/apk/app.apk",
1855 presigned: true,
1856 }
1857
1858 android_app_import {
1859 name: "bar",
1860 apk: "prebuilts/apk/app.apk",
1861 presigned: true,
1862 filename: "bar_sample.apk"
1863 }
1864 `)
1865
1866 testCases := []struct {
1867 name string
1868 expected string
1869 }{
1870 {
1871 name: "foo",
1872 expected: "foo.apk",
1873 },
1874 {
1875 name: "bar",
1876 expected: "bar_sample.apk",
1877 },
1878 }
1879
1880 for _, test := range testCases {
1881 variant := ctx.ModuleForTests(test.name, "android_common")
1882 if variant.MaybeOutput(test.expected).Rule == nil {
1883 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
1884 }
1885
1886 a := variant.Module().(*AndroidAppImport)
1887 expectedValues := []string{test.expected}
1888 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09001889 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07001890 if !reflect.DeepEqual(actualValues, expectedValues) {
1891 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
1892 actualValues, expectedValues)
1893 }
1894 }
1895}
1896
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001897func TestAndroidAppImport_ArchVariants(t *testing.T) {
1898 // The test config's target arch is ARM64.
1899 testCases := []struct {
1900 name string
1901 bp string
1902 expected string
1903 }{
1904 {
1905 name: "matching arch",
1906 bp: `
1907 android_app_import {
1908 name: "foo",
1909 apk: "prebuilts/apk/app.apk",
1910 arch: {
1911 arm64: {
1912 apk: "prebuilts/apk/app_arm64.apk",
1913 },
1914 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001915 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001916 dex_preopt: {
1917 enabled: true,
1918 },
1919 }
1920 `,
1921 expected: "prebuilts/apk/app_arm64.apk",
1922 },
1923 {
1924 name: "no matching arch",
1925 bp: `
1926 android_app_import {
1927 name: "foo",
1928 apk: "prebuilts/apk/app.apk",
1929 arch: {
1930 arm: {
1931 apk: "prebuilts/apk/app_arm.apk",
1932 },
1933 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001934 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001935 dex_preopt: {
1936 enabled: true,
1937 },
1938 }
1939 `,
1940 expected: "prebuilts/apk/app.apk",
1941 },
1942 }
1943
1944 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
1945 for _, test := range testCases {
1946 ctx, _ := testJava(t, test.bp)
1947
1948 variant := ctx.ModuleForTests("foo", "android_common")
1949 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
1950 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
1951 if len(matches) != 2 {
1952 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
1953 }
1954 if test.expected != matches[1] {
1955 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
1956 }
1957 }
1958}
1959
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001960func TestAndroidTestImport(t *testing.T) {
1961 ctx, config := testJava(t, `
1962 android_test_import {
1963 name: "foo",
1964 apk: "prebuilts/apk/app.apk",
1965 presigned: true,
1966 data: [
1967 "testdata/data",
1968 ],
1969 }
1970 `)
1971
1972 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
1973
1974 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09001975 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001976 expected := []string{"tests"}
1977 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
1978 if !reflect.DeepEqual(expected, actual) {
1979 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
1980 }
1981 expected = []string{"testdata/data:testdata/data"}
1982 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
1983 if !reflect.DeepEqual(expected, actual) {
1984 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
1985 }
1986}
1987
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001988func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
1989 ctx, _ := testJava(t, `
1990 android_test_import {
1991 name: "foo",
1992 apk: "prebuilts/apk/app.apk",
1993 certificate: "cert/new_cert",
1994 data: [
1995 "testdata/data",
1996 ],
1997 }
1998
1999 android_test_import {
2000 name: "foo_presigned",
2001 apk: "prebuilts/apk/app.apk",
2002 presigned: true,
2003 data: [
2004 "testdata/data",
2005 ],
2006 }
2007 `)
2008
2009 variant := ctx.ModuleForTests("foo", "android_common")
2010 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
2011 if !strings.HasPrefix(jniRule, "if (zipinfo") {
2012 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
2013 }
2014
2015 variant = ctx.ModuleForTests("foo_presigned", "android_common")
2016 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
2017 if jniRule != android.Cp.String() {
2018 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
2019 }
2020}
2021
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002022func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002023 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002024 cc_library {
2025 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08002026 sdk_version: "current",
2027 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002028 }
2029
2030 android_test {
2031 name: "stl",
2032 jni_libs: ["libjni"],
2033 compile_multilib: "both",
2034 sdk_version: "current",
2035 stl: "c++_shared",
2036 }
2037
2038 android_test {
2039 name: "system",
2040 jni_libs: ["libjni"],
2041 compile_multilib: "both",
2042 sdk_version: "current",
2043 }
2044 `)
2045
2046 testCases := []struct {
2047 name string
2048 jnis []string
2049 }{
2050 {"stl",
2051 []string{
2052 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07002053 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07002054 },
2055 },
2056 {"system",
2057 []string{
2058 "libjni.so",
2059 },
2060 },
2061 }
2062
2063 for _, test := range testCases {
2064 t.Run(test.name, func(t *testing.T) {
2065 app := ctx.ModuleForTests(test.name, "android_common")
2066 jniLibZip := app.Output("jnilibs.zip")
2067 var jnis []string
2068 args := strings.Fields(jniLibZip.Args["jarArgs"])
2069 for i := 0; i < len(args); i++ {
2070 if args[i] == "-f" {
2071 jnis = append(jnis, args[i+1])
2072 i += 1
2073 }
2074 }
2075 jnisJoined := strings.Join(jnis, " ")
2076 for _, jni := range test.jnis {
2077 if !strings.Contains(jnisJoined, jni) {
2078 t.Errorf("missing jni %q in %q", jni, jnis)
2079 }
2080 }
2081 })
2082 }
2083}
Colin Cross50ddcc42019-05-16 12:28:22 -07002084
2085func TestUsesLibraries(t *testing.T) {
2086 bp := `
2087 java_sdk_library {
2088 name: "foo",
2089 srcs: ["a.java"],
2090 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002091 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002092 }
2093
2094 java_sdk_library {
2095 name: "bar",
2096 srcs: ["a.java"],
2097 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002098 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002099 }
2100
2101 android_app {
2102 name: "app",
2103 srcs: ["a.java"],
2104 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002105 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002106 optional_uses_libs: [
2107 "bar",
2108 "baz",
2109 ],
2110 }
2111
2112 android_app_import {
2113 name: "prebuilt",
2114 apk: "prebuilts/apk/app.apk",
2115 certificate: "platform",
2116 uses_libs: ["foo"],
2117 optional_uses_libs: [
2118 "bar",
2119 "baz",
2120 ],
2121 }
2122 `
2123
Colin Cross98be1bb2019-12-13 20:41:13 -08002124 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002125 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2126
Colin Cross98be1bb2019-12-13 20:41:13 -08002127 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002128
2129 run(t, ctx, config)
2130
2131 app := ctx.ModuleForTests("app", "android_common")
2132 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2133
2134 // Test that all libraries are verified
2135 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2136 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2137 t.Errorf("wanted %q in %q", w, cmd)
2138 }
2139
2140 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2141 t.Errorf("wanted %q in %q", w, cmd)
2142 }
2143
2144 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2145
2146 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2147 t.Errorf("wanted %q in %q", w, cmd)
2148 }
2149
2150 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2151 t.Errorf("wanted %q in %q", w, cmd)
2152 }
2153
2154 // Test that only present libraries are preopted
2155 cmd = app.Rule("dexpreopt").RuleParams.Command
2156
2157 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2158 t.Errorf("wanted %q in %q", w, cmd)
2159 }
2160
2161 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2162
2163 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2164 t.Errorf("wanted %q in %q", w, cmd)
2165 }
2166}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002167
2168func TestCodelessApp(t *testing.T) {
2169 testCases := []struct {
2170 name string
2171 bp string
2172 noCode bool
2173 }{
2174 {
2175 name: "normal",
2176 bp: `
2177 android_app {
2178 name: "foo",
2179 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002180 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002181 }
2182 `,
2183 noCode: false,
2184 },
2185 {
2186 name: "app without sources",
2187 bp: `
2188 android_app {
2189 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002190 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002191 }
2192 `,
2193 noCode: true,
2194 },
2195 {
2196 name: "app with libraries",
2197 bp: `
2198 android_app {
2199 name: "foo",
2200 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002201 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002202 }
2203
2204 java_library {
2205 name: "lib",
2206 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002207 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002208 }
2209 `,
2210 noCode: false,
2211 },
2212 {
2213 name: "app with sourceless libraries",
2214 bp: `
2215 android_app {
2216 name: "foo",
2217 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002218 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002219 }
2220
2221 java_library {
2222 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002223 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002224 }
2225 `,
2226 // TODO(jungjw): this should probably be true
2227 noCode: false,
2228 },
2229 }
2230
2231 for _, test := range testCases {
2232 t.Run(test.name, func(t *testing.T) {
2233 ctx := testApp(t, test.bp)
2234
2235 foo := ctx.ModuleForTests("foo", "android_common")
2236 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2237 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2238 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2239 }
2240 })
2241 }
2242}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002243
2244func TestEmbedNotice(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002245 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002246 android_app {
2247 name: "foo",
2248 srcs: ["a.java"],
2249 static_libs: ["javalib"],
2250 jni_libs: ["libjni"],
2251 notice: "APP_NOTICE",
2252 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002253 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002254 }
2255
2256 // No embed_notice flag
2257 android_app {
2258 name: "bar",
2259 srcs: ["a.java"],
2260 jni_libs: ["libjni"],
2261 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002262 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002263 }
2264
2265 // No NOTICE files
2266 android_app {
2267 name: "baz",
2268 srcs: ["a.java"],
2269 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002270 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002271 }
2272
2273 cc_library {
2274 name: "libjni",
2275 system_shared_libs: [],
2276 stl: "none",
2277 notice: "LIB_NOTICE",
Colin Cross1c93c292020-02-15 10:38:00 -08002278 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002279 }
2280
2281 java_library {
2282 name: "javalib",
2283 srcs: [
2284 ":gen",
2285 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002286 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002287 }
2288
2289 genrule {
2290 name: "gen",
2291 tools: ["gentool"],
2292 out: ["gen.java"],
2293 notice: "GENRULE_NOTICE",
2294 }
2295
2296 java_binary_host {
2297 name: "gentool",
2298 srcs: ["b.java"],
2299 notice: "TOOL_NOTICE",
2300 }
2301 `)
2302
2303 // foo has NOTICE files to process, and embed_notices is true.
2304 foo := ctx.ModuleForTests("foo", "android_common")
2305 // verify merge notices rule.
2306 mergeNotices := foo.Rule("mergeNoticesRule")
2307 noticeInputs := mergeNotices.Inputs.Strings()
2308 // TOOL_NOTICE should be excluded as it's a host module.
2309 if len(mergeNotices.Inputs) != 3 {
2310 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2311 }
2312 if !inList("APP_NOTICE", noticeInputs) {
2313 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2314 }
2315 if !inList("LIB_NOTICE", noticeInputs) {
2316 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2317 }
2318 if !inList("GENRULE_NOTICE", noticeInputs) {
2319 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2320 }
2321 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2322 res := foo.Output("package-res.apk")
2323 aapt2Flags := res.Args["flags"]
2324 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2325 if !strings.Contains(aapt2Flags, e) {
2326 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2327 }
2328
2329 // bar has NOTICE files to process, but embed_notices is not set.
2330 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002331 res = bar.Output("package-res.apk")
2332 aapt2Flags = res.Args["flags"]
2333 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2334 if strings.Contains(aapt2Flags, e) {
2335 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002336 }
2337
2338 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2339 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002340 res = baz.Output("package-res.apk")
2341 aapt2Flags = res.Args["flags"]
2342 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2343 if strings.Contains(aapt2Flags, e) {
2344 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002345 }
2346}
Colin Cross53a87f52019-06-25 13:35:30 -07002347
2348func TestUncompressDex(t *testing.T) {
2349 testCases := []struct {
2350 name string
2351 bp string
2352
2353 uncompressedPlatform bool
2354 uncompressedUnbundled 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",
Colin Cross53a87f52019-06-25 13:35:30 -07002363 }
2364 `,
2365 uncompressedPlatform: true,
2366 uncompressedUnbundled: false,
2367 },
2368 {
2369 name: "use_embedded_dex",
2370 bp: `
2371 android_app {
2372 name: "foo",
2373 use_embedded_dex: true,
2374 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002375 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002376 }
2377 `,
2378 uncompressedPlatform: true,
2379 uncompressedUnbundled: true,
2380 },
2381 {
2382 name: "privileged",
2383 bp: `
2384 android_app {
2385 name: "foo",
2386 privileged: true,
2387 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002388 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002389 }
2390 `,
2391 uncompressedPlatform: true,
2392 uncompressedUnbundled: true,
2393 },
2394 }
2395
2396 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2397 t.Helper()
2398
Colin Cross98be1bb2019-12-13 20:41:13 -08002399 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002400 if unbundled {
2401 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2402 }
2403
Colin Cross98be1bb2019-12-13 20:41:13 -08002404 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002405
2406 run(t, ctx, config)
2407
2408 foo := ctx.ModuleForTests("foo", "android_common")
2409 dex := foo.Rule("r8")
2410 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2411 aligned := foo.MaybeRule("zipalign").Rule != nil
2412
2413 if uncompressedInDexJar != want {
2414 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2415 }
2416
2417 if aligned != want {
2418 t.Errorf("want aligned %v, got %v", want, aligned)
2419 }
2420 }
2421
2422 for _, tt := range testCases {
2423 t.Run(tt.name, func(t *testing.T) {
2424 t.Run("platform", func(t *testing.T) {
2425 test(t, tt.bp, tt.uncompressedPlatform, false)
2426 })
2427 t.Run("unbundled", func(t *testing.T) {
2428 test(t, tt.bp, tt.uncompressedUnbundled, true)
2429 })
2430 })
2431 }
2432}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002433
2434func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2435 if expectedValue != "" {
2436 expectedFlag := "--" + flagName + " " + expectedValue
2437 if !strings.Contains(aapt2Flags, expectedFlag) {
2438 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2439 }
2440 } else {
2441 unexpectedFlag := "--" + flagName
2442 if strings.Contains(aapt2Flags, unexpectedFlag) {
2443 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2444 }
2445 }
2446}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002447
2448func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungca095d72020-04-09 16:15:30 -07002449 fs := map[string][]byte{
2450 "baz/res/res/values/strings.xml": nil,
2451 "bar/res/res/values/strings.xml": nil,
2452 }
2453 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002454 runtime_resource_overlay {
2455 name: "foo",
2456 certificate: "platform",
2457 product_specific: true,
Jaewoong Jungca095d72020-04-09 16:15:30 -07002458 static_libs: ["bar"],
2459 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002460 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002461 }
2462
2463 runtime_resource_overlay {
2464 name: "foo_themed",
2465 certificate: "platform",
2466 product_specific: true,
2467 theme: "faza",
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002468 overrides: ["foo"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002469 }
Jaewoong Jungca095d72020-04-09 16:15:30 -07002470
2471 android_library {
2472 name: "bar",
2473 resource_dirs: ["bar/res"],
2474 }
2475
2476 android_app {
2477 name: "baz",
2478 sdk_version: "current",
2479 resource_dirs: ["baz/res"],
2480 }
2481 `
2482 config := testAppConfig(nil, bp, fs)
2483 ctx := testContext()
2484 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002485
2486 m := ctx.ModuleForTests("foo", "android_common")
2487
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002488 // Check AAPT2 link flags.
2489 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2490 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2491 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2492 if len(absentFlags) > 0 {
2493 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2494 }
2495
Jaewoong Jungca095d72020-04-09 16:15:30 -07002496 // Check overlay.list output for static_libs dependency.
2497 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
2498 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
2499 if !inList(staticLibPackage, overlayList) {
2500 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
2501 }
2502
2503 // Check AAPT2 link flags for resource_libs dependency.
2504 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
2505 if !strings.Contains(aapt2Flags, resourceLibFlag) {
2506 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
2507 }
2508
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002509 // Check cert signing flag.
2510 signedApk := m.Output("signed/foo.apk")
2511 signingFlag := signedApk.Args["certificates"]
2512 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2513 if expected != signingFlag {
2514 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2515 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002516 androidMkEntries := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2517 path := androidMkEntries.EntryMap["LOCAL_CERTIFICATE"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002518 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
2519 if !reflect.DeepEqual(path, expectedPath) {
2520 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
2521 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002522
2523 // Check device location.
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002524 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002525 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002526 if !reflect.DeepEqual(path, expectedPath) {
2527 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2528 }
2529
2530 // A themed module has a different device location
2531 m = ctx.ModuleForTests("foo_themed", "android_common")
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002532 androidMkEntries = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0]
2533 path = androidMkEntries.EntryMap["LOCAL_MODULE_PATH"]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002534 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
2535 if !reflect.DeepEqual(path, expectedPath) {
2536 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2537 }
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07002538
2539 overrides := androidMkEntries.EntryMap["LOCAL_OVERRIDES_PACKAGES"]
2540 expectedOverrides := []string{"foo"}
2541 if !reflect.DeepEqual(overrides, expectedOverrides) {
2542 t.Errorf("Unexpected LOCAL_OVERRIDES_PACKAGES value: %v, expected: %v", overrides, expectedOverrides)
2543 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002544}
Jaewoong Jung062ed7e2020-04-26 15:10:51 -07002545
2546func TestRuntimeResourceOverlay_JavaDefaults(t *testing.T) {
2547 ctx, config := testJava(t, `
2548 java_defaults {
2549 name: "rro_defaults",
2550 theme: "default_theme",
2551 product_specific: true,
2552 aaptflags: ["--keep-raw-values"],
2553 }
2554
2555 runtime_resource_overlay {
2556 name: "foo_with_defaults",
2557 defaults: ["rro_defaults"],
2558 }
2559
2560 runtime_resource_overlay {
2561 name: "foo_barebones",
2562 }
2563 `)
2564
2565 //
2566 // RRO module with defaults
2567 //
2568 m := ctx.ModuleForTests("foo_with_defaults", "android_common")
2569
2570 // Check AAPT2 link flags.
2571 aapt2Flags := strings.Split(m.Output("package-res.apk").Args["flags"], " ")
2572 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2573 absentFlags := android.RemoveListFromList(expectedFlags, aapt2Flags)
2574 if len(absentFlags) > 0 {
2575 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2576 }
2577
2578 // Check device location.
2579 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2580 expectedPath := []string{"/tmp/target/product/test_device/product/overlay/default_theme"}
2581 if !reflect.DeepEqual(path, expectedPath) {
2582 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %q, expected: %q", path, expectedPath)
2583 }
2584
2585 //
2586 // RRO module without defaults
2587 //
2588 m = ctx.ModuleForTests("foo_barebones", "android_common")
2589
2590 // Check AAPT2 link flags.
2591 aapt2Flags = strings.Split(m.Output("package-res.apk").Args["flags"], " ")
2592 unexpectedFlags := "--keep-raw-values"
2593 if inList(unexpectedFlags, aapt2Flags) {
2594 t.Errorf("unexpected value, %q is present in aapt2 link flags, %q", unexpectedFlags, aapt2Flags)
2595 }
2596
2597 // Check device location.
2598 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2599 expectedPath = []string{"/tmp/target/product/test_device/system/overlay"}
2600 if !reflect.DeepEqual(path, expectedPath) {
2601 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2602 }
2603}