blob: 43696dbb8607c7fd68d2b2beb92997c09ae57a8c [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
Colin Cross0ddae7f2019-02-07 15:30:01 -0800267func TestResourceDirs(t *testing.T) {
268 testCases := []struct {
269 name string
270 prop string
271 resources []string
272 }{
273 {
274 name: "no resource_dirs",
275 prop: "",
276 resources: []string{"res/res/values/strings.xml"},
277 },
278 {
279 name: "resource_dirs",
280 prop: `resource_dirs: ["res"]`,
281 resources: []string{"res/res/values/strings.xml"},
282 },
283 {
284 name: "empty resource_dirs",
285 prop: `resource_dirs: []`,
286 resources: nil,
287 },
288 }
289
290 fs := map[string][]byte{
291 "res/res/values/strings.xml": nil,
292 }
293
294 bp := `
295 android_app {
296 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900297 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800298 %s
299 }
300 `
301
302 for _, testCase := range testCases {
303 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800304 config := testConfig(nil, fmt.Sprintf(bp, testCase.prop), fs)
305 ctx := testContext()
Colin Cross0ddae7f2019-02-07 15:30:01 -0800306 run(t, ctx, config)
307
308 module := ctx.ModuleForTests("foo", "android_common")
309 resourceList := module.MaybeOutput("aapt2/res.list")
310
311 var resources []string
312 if resourceList.Rule != nil {
313 for _, compiledResource := range resourceList.Inputs.Strings() {
314 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
315 }
316 }
317
318 if !reflect.DeepEqual(resources, testCase.resources) {
319 t.Errorf("expected resource files %q, got %q",
320 testCase.resources, resources)
321 }
322 })
323 }
324}
325
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800326func TestLibraryAssets(t *testing.T) {
327 bp := `
328 android_app {
329 name: "foo",
330 sdk_version: "current",
331 static_libs: ["lib1", "lib2", "lib3"],
332 }
333
334 android_library {
335 name: "lib1",
336 sdk_version: "current",
337 asset_dirs: ["assets_a"],
338 }
339
340 android_library {
341 name: "lib2",
342 sdk_version: "current",
343 }
344
345 android_library {
346 name: "lib3",
347 sdk_version: "current",
348 static_libs: ["lib4"],
349 }
350
351 android_library {
352 name: "lib4",
353 sdk_version: "current",
354 asset_dirs: ["assets_b"],
355 }
356 `
357
358 testCases := []struct {
359 name string
360 assetFlag string
361 assetPackages []string
362 }{
363 {
364 name: "foo",
365 // lib1 has its own asset. lib3 doesn't have any, but provides lib4's transitively.
366 assetPackages: []string{
367 buildDir + "/.intermediates/foo/android_common/aapt2/package-res.apk",
368 buildDir + "/.intermediates/lib1/android_common/assets.zip",
369 buildDir + "/.intermediates/lib3/android_common/assets.zip",
370 },
371 },
372 {
373 name: "lib1",
374 assetFlag: "-A assets_a",
375 },
376 {
377 name: "lib2",
378 },
379 {
380 name: "lib3",
381 assetPackages: []string{
382 buildDir + "/.intermediates/lib3/android_common/aapt2/package-res.apk",
383 buildDir + "/.intermediates/lib4/android_common/assets.zip",
384 },
385 },
386 {
387 name: "lib4",
388 assetFlag: "-A assets_b",
389 },
390 }
391 ctx := testApp(t, bp)
392
393 for _, test := range testCases {
394 t.Run(test.name, func(t *testing.T) {
395 m := ctx.ModuleForTests(test.name, "android_common")
396
397 // Check asset flag in aapt2 link flags
398 var aapt2link android.TestingBuildParams
399 if len(test.assetPackages) > 0 {
400 aapt2link = m.Output("aapt2/package-res.apk")
401 } else {
402 aapt2link = m.Output("package-res.apk")
403 }
404 aapt2Flags := aapt2link.Args["flags"]
405 if test.assetFlag != "" {
406 if !strings.Contains(aapt2Flags, test.assetFlag) {
407 t.Errorf("Can't find asset flag %q in aapt2 link flags %q", test.assetFlag, aapt2Flags)
408 }
409 } else {
410 if strings.Contains(aapt2Flags, " -A ") {
411 t.Errorf("aapt2 link flags %q contain unexpected asset flag", aapt2Flags)
412 }
413 }
414
415 // Check asset merge rule.
416 if len(test.assetPackages) > 0 {
417 mergeAssets := m.Output("package-res.apk")
418 if !reflect.DeepEqual(test.assetPackages, mergeAssets.Inputs.Strings()) {
419 t.Errorf("Unexpected mergeAssets inputs: %v, expected: %v",
420 mergeAssets.Inputs.Strings(), test.assetPackages)
421 }
422 }
423 })
424 }
425}
426
Colin Crossbec85302019-02-13 13:15:46 -0800427func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800428 testCases := []struct {
429 name string
430 enforceRROTargets []string
431 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800432 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800433 overlayFiles map[string][]string
434 rroDirs map[string][]string
435 }{
436 {
437 name: "no RRO",
438 enforceRROTargets: nil,
439 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800440 resourceFiles: map[string][]string{
441 "foo": nil,
442 "bar": {"bar/res/res/values/strings.xml"},
443 "lib": nil,
444 "lib2": {"lib2/res/res/values/strings.xml"},
445 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800446 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800447 "foo": {
448 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800449 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000450 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800451 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800452 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
453 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000454 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800455 },
Colin Crossbec85302019-02-13 13:15:46 -0800456 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800457 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
458 "device/vendor/blah/overlay/bar/res/values/strings.xml",
459 },
Colin Crossbec85302019-02-13 13:15:46 -0800460 "lib": {
461 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
462 "lib/res/res/values/strings.xml",
463 "device/vendor/blah/overlay/lib/res/values/strings.xml",
464 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800465 },
466 rroDirs: map[string][]string{
467 "foo": nil,
468 "bar": nil,
469 },
470 },
471 {
472 name: "enforce RRO on foo",
473 enforceRROTargets: []string{"foo"},
474 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800475 resourceFiles: map[string][]string{
476 "foo": nil,
477 "bar": {"bar/res/res/values/strings.xml"},
478 "lib": nil,
479 "lib2": {"lib2/res/res/values/strings.xml"},
480 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800481 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800482 "foo": {
483 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800484 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000485 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800486 "foo/res/res/values/strings.xml",
487 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
488 },
Colin Crossbec85302019-02-13 13:15:46 -0800489 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800490 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
491 "device/vendor/blah/overlay/bar/res/values/strings.xml",
492 },
Colin Crossbec85302019-02-13 13:15:46 -0800493 "lib": {
494 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
495 "lib/res/res/values/strings.xml",
496 "device/vendor/blah/overlay/lib/res/values/strings.xml",
497 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800498 },
Colin Crossc1c37552019-01-31 11:42:41 -0800499
Colin Cross5c4791c2019-02-01 11:44:44 -0800500 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800501 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000502 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800503 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
504 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000505 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800506 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800507 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800508 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800509 },
510 },
511 {
512 name: "enforce RRO on all",
513 enforceRROTargets: []string{"*"},
514 enforceRROExcludedOverlays: []string{
515 // Excluding specific apps/res directories also allowed.
516 "device/vendor/blah/static_overlay/foo",
517 "device/vendor/blah/static_overlay/bar/res",
518 },
Colin Crossbec85302019-02-13 13:15:46 -0800519 resourceFiles: map[string][]string{
520 "foo": nil,
521 "bar": {"bar/res/res/values/strings.xml"},
522 "lib": nil,
523 "lib2": {"lib2/res/res/values/strings.xml"},
524 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800525 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800526 "foo": {
527 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800528 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000529 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800530 "foo/res/res/values/strings.xml",
531 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
532 },
Colin Crossbec85302019-02-13 13:15:46 -0800533 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
534 "lib": {
535 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
536 "lib/res/res/values/strings.xml",
537 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800538 },
539 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800540 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000541 "device:device/vendor/blah/overlay/foo/res",
542 "product:product/vendor/blah/overlay/foo/res",
543 // Lib dep comes after the direct deps
544 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800545 },
Anton Hansson53c88442019-03-18 15:53:16 +0000546 "bar": {"device:device/vendor/blah/overlay/bar/res"},
547 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800548 },
549 },
550 }
551
Anton Hansson53c88442019-03-18 15:53:16 +0000552 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800553 "device/vendor/blah/overlay",
554 "device/vendor/blah/overlay2",
555 "device/vendor/blah/static_overlay",
556 }
557
Anton Hansson53c88442019-03-18 15:53:16 +0000558 productResourceOverlays := []string{
559 "product/vendor/blah/overlay",
560 }
561
Colin Cross890ff552017-11-30 20:13:19 -0800562 fs := map[string][]byte{
563 "foo/res/res/values/strings.xml": nil,
564 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800565 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800566 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800567 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
568 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800569 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800570 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
571 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
572 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000573 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800574 }
575
576 bp := `
577 android_app {
578 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900579 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800580 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000581 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800582 }
583
584 android_app {
585 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900586 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800587 resource_dirs: ["bar/res"],
588 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800589
590 android_library {
591 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900592 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800593 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800594 static_libs: ["lib2"],
595 }
596
597 android_library {
598 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900599 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800600 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800601 }
Anton Hansson53c88442019-03-18 15:53:16 +0000602
603 // This library has the same resources as lib (should not lead to dupe RROs)
604 android_library {
605 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900606 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000607 resource_dirs: ["lib/res"]
608 }
Colin Cross890ff552017-11-30 20:13:19 -0800609 `
610
Colin Cross5c4791c2019-02-01 11:44:44 -0800611 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800612 t.Run(testCase.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -0800613 config := testAppConfig(nil, bp, fs)
Anton Hansson53c88442019-03-18 15:53:16 +0000614 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
615 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800616 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800617 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800618 }
619 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800620 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800621 }
622
Colin Cross98be1bb2019-12-13 20:41:13 -0800623 ctx := testContext()
Colin Cross890ff552017-11-30 20:13:19 -0800624 run(t, ctx, config)
625
Colin Crossbec85302019-02-13 13:15:46 -0800626 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
627 for _, o := range list {
628 res := module.MaybeOutput(o)
629 if res.Rule != nil {
630 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
631 // verify the inputs to the .arsc.flat rule.
632 files = append(files, res.Inputs.Strings()...)
633 } else {
634 // Otherwise, verify the full path to the output of the other module
635 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000636 }
Colin Cross890ff552017-11-30 20:13:19 -0800637 }
Colin Crossbec85302019-02-13 13:15:46 -0800638 return files
Colin Cross890ff552017-11-30 20:13:19 -0800639 }
640
Colin Crossbec85302019-02-13 13:15:46 -0800641 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
642 module := ctx.ModuleForTests(moduleName, "android_common")
643 resourceList := module.MaybeOutput("aapt2/res.list")
644 if resourceList.Rule != nil {
645 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000646 }
Colin Crossbec85302019-02-13 13:15:46 -0800647 overlayList := module.MaybeOutput("aapt2/overlay.list")
648 if overlayList.Rule != nil {
649 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
650 }
651
Anton Hansson53c88442019-03-18 15:53:16 +0000652 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
653 var prefix string
654 if d.overlayType == device {
655 prefix = "device:"
656 } else if d.overlayType == product {
657 prefix = "product:"
658 } else {
659 t.Fatalf("Unexpected overlayType %d", d.overlayType)
660 }
661 rroDirs = append(rroDirs, prefix+d.path.String())
662 }
Colin Crossbec85302019-02-13 13:15:46 -0800663
664 return resourceFiles, overlayFiles, rroDirs
665 }
666
667 modules := []string{"foo", "bar", "lib", "lib2"}
668 for _, module := range modules {
669 resourceFiles, overlayFiles, rroDirs := getResources(module)
670
671 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
672 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
673 module, testCase.resourceFiles[module], resourceFiles)
674 }
675 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
676 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
677 module, testCase.overlayFiles[module], overlayFiles)
678 }
679 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +0000680 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -0800681 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +0000682 }
Colin Cross890ff552017-11-30 20:13:19 -0800683 }
Colin Cross890ff552017-11-30 20:13:19 -0800684 })
685 }
686}
Colin Crossd09b0b62018-04-18 11:06:47 -0700687
688func TestAppSdkVersion(t *testing.T) {
689 testCases := []struct {
690 name string
691 sdkVersion string
692 platformSdkInt int
693 platformSdkCodename string
694 platformSdkFinal bool
695 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +0900696 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -0700697 }{
698 {
699 name: "current final SDK",
700 sdkVersion: "current",
701 platformSdkInt: 27,
702 platformSdkCodename: "REL",
703 platformSdkFinal: true,
704 expectedMinSdkVersion: "27",
705 },
706 {
707 name: "current non-final SDK",
708 sdkVersion: "current",
709 platformSdkInt: 27,
710 platformSdkCodename: "OMR1",
711 platformSdkFinal: false,
712 expectedMinSdkVersion: "OMR1",
713 },
714 {
715 name: "default final SDK",
716 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900717 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700718 platformSdkInt: 27,
719 platformSdkCodename: "REL",
720 platformSdkFinal: true,
721 expectedMinSdkVersion: "27",
722 },
723 {
724 name: "default non-final SDK",
725 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900726 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700727 platformSdkInt: 27,
728 platformSdkCodename: "OMR1",
729 platformSdkFinal: false,
730 expectedMinSdkVersion: "OMR1",
731 },
732 {
733 name: "14",
734 sdkVersion: "14",
735 expectedMinSdkVersion: "14",
736 },
737 }
738
739 for _, moduleType := range []string{"android_app", "android_library"} {
740 for _, test := range testCases {
741 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +0900742 platformApiProp := ""
743 if test.platformApis {
744 platformApiProp = "platform_apis: true,"
745 }
Colin Crossd09b0b62018-04-18 11:06:47 -0700746 bp := fmt.Sprintf(`%s {
747 name: "foo",
748 srcs: ["a.java"],
749 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900750 %s
751 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -0700752
Colin Cross98be1bb2019-12-13 20:41:13 -0800753 config := testAppConfig(nil, bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -0700754 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
755 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
756 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
757
Colin Cross98be1bb2019-12-13 20:41:13 -0800758 ctx := testContext()
Colin Crossd09b0b62018-04-18 11:06:47 -0700759
760 run(t, ctx, config)
761
762 foo := ctx.ModuleForTests("foo", "android_common")
763 link := foo.Output("package-res.apk")
764 linkFlags := strings.Split(link.Args["flags"], " ")
765 min := android.IndexList("--min-sdk-version", linkFlags)
766 target := android.IndexList("--target-sdk-version", linkFlags)
767
768 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
769 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
770 }
771
772 gotMinSdkVersion := linkFlags[min+1]
773 gotTargetSdkVersion := linkFlags[target+1]
774
775 if gotMinSdkVersion != test.expectedMinSdkVersion {
776 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
777 test.expectedMinSdkVersion, gotMinSdkVersion)
778 }
779
780 if gotTargetSdkVersion != test.expectedMinSdkVersion {
781 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
782 test.expectedMinSdkVersion, gotTargetSdkVersion)
783 }
784 })
785 }
786 }
787}
Colin Crossa4f08812018-10-02 22:03:40 -0700788
Paul Duffin50c217c2019-06-12 13:25:22 +0100789func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700790 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +0100791 cc_library {
792 name: "libjni",
793 system_shared_libs: [],
Colin Crossc511bc52020-04-07 16:50:32 +0000794 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100795 stl: "none",
796 }
797
798 android_test {
799 name: "test",
800 sdk_version: "core_platform",
801 jni_libs: ["libjni"],
802 }
803
804 android_test {
805 name: "test_first",
806 sdk_version: "core_platform",
807 compile_multilib: "first",
808 jni_libs: ["libjni"],
809 }
810
811 android_test {
812 name: "test_both",
813 sdk_version: "core_platform",
814 compile_multilib: "both",
815 jni_libs: ["libjni"],
816 }
817
818 android_test {
819 name: "test_32",
820 sdk_version: "core_platform",
821 compile_multilib: "32",
822 jni_libs: ["libjni"],
823 }
824
825 android_test {
826 name: "test_64",
827 sdk_version: "core_platform",
828 compile_multilib: "64",
829 jni_libs: ["libjni"],
830 }
831 `)
832
833 testCases := []struct {
834 name string
835 abis []string
836 }{
837 {"test", []string{"arm64-v8a"}},
838 {"test_first", []string{"arm64-v8a"}},
839 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
840 {"test_32", []string{"armeabi-v7a"}},
841 {"test_64", []string{"arm64-v8a"}},
842 }
843
844 for _, test := range testCases {
845 t.Run(test.name, func(t *testing.T) {
846 app := ctx.ModuleForTests(test.name, "android_common")
847 jniLibZip := app.Output("jnilibs.zip")
848 var abis []string
849 args := strings.Fields(jniLibZip.Args["jarArgs"])
850 for i := 0; i < len(args); i++ {
851 if args[i] == "-P" {
852 abis = append(abis, filepath.Base(args[i+1]))
853 i++
854 }
855 }
856 if !reflect.DeepEqual(abis, test.abis) {
857 t.Errorf("want abis %v, got %v", test.abis, abis)
858 }
859 })
860 }
861}
862
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900863func TestAppSdkVersionByPartition(t *testing.T) {
864 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
865 android_app {
866 name: "foo",
867 srcs: ["a.java"],
868 vendor: true,
869 platform_apis: true,
870 }
871 `)
872
873 testJava(t, `
874 android_app {
875 name: "bar",
876 srcs: ["b.java"],
877 platform_apis: true,
878 }
879 `)
880
881 for _, enforce := range []bool{true, false} {
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900882 bp := `
883 android_app {
884 name: "foo",
885 srcs: ["a.java"],
886 product_specific: true,
887 platform_apis: true,
888 }
889 `
Colin Cross98be1bb2019-12-13 20:41:13 -0800890
891 config := testAppConfig(nil, bp, nil)
892 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900893 if enforce {
Colin Cross98be1bb2019-12-13 20:41:13 -0800894 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 +0900895 } else {
Colin Cross98be1bb2019-12-13 20:41:13 -0800896 testJavaWithConfig(t, config)
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900897 }
898 }
899}
900
Paul Duffin50c217c2019-06-12 13:25:22 +0100901func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700902 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +0100903 cc_library {
904 name: "libjni",
905 system_shared_libs: [],
906 stl: "none",
Colin Cross094cde42020-02-15 10:38:00 -0800907 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100908 }
909
910 android_app {
911 name: "app",
912 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900913 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100914 }
915
916 android_app {
917 name: "app_noembed",
918 jni_libs: ["libjni"],
919 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +0900920 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100921 }
922
923 android_app {
924 name: "app_embed",
925 jni_libs: ["libjni"],
926 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +0900927 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100928 }
929
930 android_test {
931 name: "test",
Colin Crossc511bc52020-04-07 16:50:32 +0000932 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100933 jni_libs: ["libjni"],
934 }
935
936 android_test {
937 name: "test_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +0000938 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100939 jni_libs: ["libjni"],
940 use_embedded_native_libs: false,
941 }
942
943 android_test_helper_app {
944 name: "test_helper",
Colin Crossc511bc52020-04-07 16:50:32 +0000945 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100946 jni_libs: ["libjni"],
947 }
948
949 android_test_helper_app {
950 name: "test_helper_noembed",
Colin Crossc511bc52020-04-07 16:50:32 +0000951 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100952 jni_libs: ["libjni"],
953 use_embedded_native_libs: false,
954 }
955 `)
956
957 testCases := []struct {
958 name string
959 packaged bool
960 compressed bool
961 }{
962 {"app", false, false},
963 {"app_noembed", false, false},
964 {"app_embed", true, false},
965 {"test", true, false},
966 {"test_noembed", true, true},
967 {"test_helper", true, false},
968 {"test_helper_noembed", true, true},
969 }
970
971 for _, test := range testCases {
972 t.Run(test.name, func(t *testing.T) {
973 app := ctx.ModuleForTests(test.name, "android_common")
974 jniLibZip := app.MaybeOutput("jnilibs.zip")
975 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
976 t.Errorf("expected jni packaged %v, got %v", w, g)
977 }
978
979 if jniLibZip.Rule != nil {
980 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
981 t.Errorf("expected jni compressed %v, got %v", w, g)
982 }
Colin Crossc511bc52020-04-07 16:50:32 +0000983
984 if !strings.Contains(jniLibZip.Implicits[0].String(), "_sdk_") {
985 t.Errorf("expected input %q to use sdk variant", jniLibZip.Implicits[0].String())
986 }
Paul Duffin50c217c2019-06-12 13:25:22 +0100987 }
988 })
989 }
Colin Cross47fa9d32019-03-26 10:51:39 -0700990}
991
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800992func TestCertificates(t *testing.T) {
993 testCases := []struct {
994 name string
995 bp string
996 certificateOverride string
997 expected string
998 }{
999 {
1000 name: "default",
1001 bp: `
1002 android_app {
1003 name: "foo",
1004 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001005 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001006 }
1007 `,
1008 certificateOverride: "",
Dan Willemsen412160e2019-04-09 21:36:26 -07001009 expected: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001010 },
1011 {
1012 name: "module certificate property",
1013 bp: `
1014 android_app {
1015 name: "foo",
1016 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001017 certificate: ":new_certificate",
1018 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001019 }
1020
1021 android_app_certificate {
1022 name: "new_certificate",
1023 certificate: "cert/new_cert",
1024 }
1025 `,
1026 certificateOverride: "",
1027 expected: "cert/new_cert.x509.pem cert/new_cert.pk8",
1028 },
1029 {
1030 name: "path certificate property",
1031 bp: `
1032 android_app {
1033 name: "foo",
1034 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001035 certificate: "expiredkey",
1036 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001037 }
1038 `,
1039 certificateOverride: "",
Dan Willemsen412160e2019-04-09 21:36:26 -07001040 expected: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001041 },
1042 {
1043 name: "certificate overrides",
1044 bp: `
1045 android_app {
1046 name: "foo",
1047 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001048 certificate: "expiredkey",
1049 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001050 }
1051
1052 android_app_certificate {
1053 name: "new_certificate",
1054 certificate: "cert/new_cert",
1055 }
1056 `,
1057 certificateOverride: "foo:new_certificate",
1058 expected: "cert/new_cert.x509.pem cert/new_cert.pk8",
1059 },
1060 }
1061
1062 for _, test := range testCases {
1063 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001064 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001065 if test.certificateOverride != "" {
1066 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
1067 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001068 ctx := testContext()
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001069
1070 run(t, ctx, config)
1071 foo := ctx.ModuleForTests("foo", "android_common")
1072
1073 signapk := foo.Output("foo.apk")
1074 signFlags := signapk.Args["certificates"]
1075 if test.expected != signFlags {
1076 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1077 }
1078 })
1079 }
1080}
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001081
Songchun Fan17d69e32020-03-24 20:32:24 -07001082func TestRequestV4SigningFlag(t *testing.T) {
1083 testCases := []struct {
1084 name string
1085 bp string
1086 expected string
1087 }{
1088 {
1089 name: "default",
1090 bp: `
1091 android_app {
1092 name: "foo",
1093 srcs: ["a.java"],
1094 sdk_version: "current",
1095 }
1096 `,
1097 expected: "",
1098 },
1099 {
1100 name: "default",
1101 bp: `
1102 android_app {
1103 name: "foo",
1104 srcs: ["a.java"],
1105 sdk_version: "current",
1106 v4_signature: false,
1107 }
1108 `,
1109 expected: "",
1110 },
1111 {
1112 name: "module certificate property",
1113 bp: `
1114 android_app {
1115 name: "foo",
1116 srcs: ["a.java"],
1117 sdk_version: "current",
1118 v4_signature: true,
1119 }
1120 `,
1121 expected: "--enable-v4",
1122 },
1123 }
1124
1125 for _, test := range testCases {
1126 t.Run(test.name, func(t *testing.T) {
1127 config := testAppConfig(nil, test.bp, nil)
1128 ctx := testContext()
1129
1130 run(t, ctx, config)
1131 foo := ctx.ModuleForTests("foo", "android_common")
1132
1133 signapk := foo.Output("foo.apk")
1134 signFlags := signapk.Args["flags"]
1135 if test.expected != signFlags {
1136 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
1137 }
1138 })
1139 }
1140}
1141
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001142func TestPackageNameOverride(t *testing.T) {
1143 testCases := []struct {
1144 name string
1145 bp string
1146 packageNameOverride string
1147 expected []string
1148 }{
1149 {
1150 name: "default",
1151 bp: `
1152 android_app {
1153 name: "foo",
1154 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001155 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001156 }
1157 `,
1158 packageNameOverride: "",
1159 expected: []string{
1160 buildDir + "/.intermediates/foo/android_common/foo.apk",
1161 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
1162 },
1163 },
1164 {
1165 name: "overridden",
1166 bp: `
1167 android_app {
1168 name: "foo",
1169 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001170 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001171 }
1172 `,
1173 packageNameOverride: "foo:bar",
1174 expected: []string{
1175 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -08001176 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001177 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
1178 },
1179 },
1180 }
1181
1182 for _, test := range testCases {
1183 t.Run(test.name, func(t *testing.T) {
Colin Cross98be1bb2019-12-13 20:41:13 -08001184 config := testAppConfig(nil, test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001185 if test.packageNameOverride != "" {
1186 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
1187 }
Colin Cross98be1bb2019-12-13 20:41:13 -08001188 ctx := testContext()
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001189
1190 run(t, ctx, config)
1191 foo := ctx.ModuleForTests("foo", "android_common")
1192
1193 outputs := foo.AllOutputs()
1194 outputMap := make(map[string]bool)
1195 for _, o := range outputs {
1196 outputMap[o] = true
1197 }
1198 for _, e := range test.expected {
1199 if _, exist := outputMap[e]; !exist {
1200 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
1201 }
1202 }
1203 })
1204 }
1205}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001206
1207func TestInstrumentationTargetOverridden(t *testing.T) {
1208 bp := `
1209 android_app {
1210 name: "foo",
1211 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001212 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001213 }
1214
1215 android_test {
1216 name: "bar",
1217 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001218 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001219 }
1220 `
Colin Cross98be1bb2019-12-13 20:41:13 -08001221 config := testAppConfig(nil, bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001222 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Colin Cross98be1bb2019-12-13 20:41:13 -08001223 ctx := testContext()
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001224
1225 run(t, ctx, config)
1226
1227 bar := ctx.ModuleForTests("bar", "android_common")
1228 res := bar.Output("package-res.apk")
1229 aapt2Flags := res.Args["flags"]
1230 e := "--rename-instrumentation-target-package org.dandroid.bp"
1231 if !strings.Contains(aapt2Flags, e) {
1232 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1233 }
1234}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001235
1236func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001237 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -08001238 android_app {
1239 name: "foo",
1240 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -07001241 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001242 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001243 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001244 }
1245
1246 override_android_app {
1247 name: "bar",
1248 base: "foo",
1249 certificate: ":new_certificate",
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001250 logging_parent: "bah",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001251 }
1252
1253 android_app_certificate {
1254 name: "new_certificate",
1255 certificate: "cert/new_cert",
1256 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001257
1258 override_android_app {
1259 name: "baz",
1260 base: "foo",
1261 package_name: "org.dandroid.bp",
1262 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001263 `)
1264
1265 expectedVariants := []struct {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001266 moduleName string
1267 variantName string
1268 apkName string
1269 apkPath string
1270 signFlag string
1271 overrides []string
1272 aaptFlag string
1273 logging_parent string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001274 }{
1275 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001276 moduleName: "foo",
1277 variantName: "android_common",
1278 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
1279 signFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1280 overrides: []string{"qux"},
1281 aaptFlag: "",
1282 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001283 },
1284 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001285 moduleName: "bar",
1286 variantName: "android_common_bar",
1287 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
1288 signFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
1289 overrides: []string{"qux", "foo"},
1290 aaptFlag: "",
1291 logging_parent: "bah",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001292 },
1293 {
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001294 moduleName: "baz",
1295 variantName: "android_common_baz",
1296 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
1297 signFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
1298 overrides: []string{"qux", "foo"},
1299 aaptFlag: "--rename-manifest-package org.dandroid.bp",
1300 logging_parent: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001301 },
1302 }
1303 for _, expected := range expectedVariants {
1304 variant := ctx.ModuleForTests("foo", expected.variantName)
1305
1306 // Check the final apk name
1307 outputs := variant.AllOutputs()
1308 expectedApkPath := buildDir + expected.apkPath
1309 found := false
1310 for _, o := range outputs {
1311 if o == expectedApkPath {
1312 found = true
1313 break
1314 }
1315 }
1316 if !found {
1317 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1318 }
1319
1320 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001321 signapk := variant.Output(expected.moduleName + ".apk")
Jaewoong Jung525443a2019-02-28 15:35:54 -08001322 signFlag := signapk.Args["certificates"]
1323 if expected.signFlag != signFlag {
1324 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.signFlag, signFlag)
1325 }
1326
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001327 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001328 mod := variant.Module().(*AndroidApp)
1329 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1330 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1331 expected.overrides, mod.appProperties.Overrides)
1332 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001333
Baligh Uddin5b16dfb2020-02-11 17:27:19 -08001334 // Test Overridable property: Logging_parent
1335 logging_parent := mod.aapt.LoggingParent
1336 if expected.logging_parent != logging_parent {
1337 t.Errorf("Incorrect overrides property value for logging parent, expected: %q, got: %q",
1338 expected.logging_parent, logging_parent)
1339 }
1340
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001341 // Check the package renaming flag, if exists.
1342 res := variant.Output("package-res.apk")
1343 aapt2Flags := res.Args["flags"]
1344 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1345 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1346 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001347 }
1348}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001349
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001350func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001351 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001352 android_app {
1353 name: "foo",
1354 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001355 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001356 }
1357
1358 override_android_app {
1359 name: "bar",
1360 base: "foo",
1361 package_name: "org.dandroid.bp",
1362 }
1363
1364 android_test {
1365 name: "baz",
1366 srcs: ["b.java"],
1367 instrumentation_for: "foo",
1368 }
1369
1370 android_test {
1371 name: "qux",
1372 srcs: ["b.java"],
1373 instrumentation_for: "bar",
1374 }
1375 `)
1376
1377 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1378 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1379 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1380 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1381 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1382 }
1383
1384 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1385 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1386 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1387 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1388 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1389 }
1390}
1391
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001392func TestOverrideAndroidTest(t *testing.T) {
1393 ctx, _ := testJava(t, `
1394 android_app {
1395 name: "foo",
1396 srcs: ["a.java"],
1397 package_name: "com.android.foo",
1398 sdk_version: "current",
1399 }
1400
1401 override_android_app {
1402 name: "bar",
1403 base: "foo",
1404 package_name: "com.android.bar",
1405 }
1406
1407 android_test {
1408 name: "foo_test",
1409 srcs: ["b.java"],
1410 instrumentation_for: "foo",
1411 }
1412
1413 override_android_test {
1414 name: "bar_test",
1415 base: "foo_test",
1416 package_name: "com.android.bar.test",
1417 instrumentation_for: "bar",
1418 instrumentation_target_package: "com.android.bar",
1419 }
1420 `)
1421
1422 expectedVariants := []struct {
1423 moduleName string
1424 variantName string
1425 apkPath string
1426 overrides []string
1427 targetVariant string
1428 packageFlag string
1429 targetPackageFlag string
1430 }{
1431 {
1432 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001433 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001434 overrides: nil,
1435 targetVariant: "android_common",
1436 packageFlag: "",
1437 targetPackageFlag: "",
1438 },
1439 {
1440 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001441 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001442 overrides: []string{"foo_test"},
1443 targetVariant: "android_common_bar",
1444 packageFlag: "com.android.bar.test",
1445 targetPackageFlag: "com.android.bar",
1446 },
1447 }
1448 for _, expected := range expectedVariants {
1449 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1450
1451 // Check the final apk name
1452 outputs := variant.AllOutputs()
1453 expectedApkPath := buildDir + expected.apkPath
1454 found := false
1455 for _, o := range outputs {
1456 if o == expectedApkPath {
1457 found = true
1458 break
1459 }
1460 }
1461 if !found {
1462 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1463 }
1464
1465 // Check if the overrides field values are correctly aggregated.
1466 mod := variant.Module().(*AndroidTest)
1467 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1468 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1469 expected.overrides, mod.appProperties.Overrides)
1470 }
1471
1472 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1473 javac := variant.Rule("javac")
1474 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1475 if !strings.Contains(javac.Args["classpath"], turbine) {
1476 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1477 }
1478
1479 // Check aapt2 flags.
1480 res := variant.Output("package-res.apk")
1481 aapt2Flags := res.Args["flags"]
1482 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1483 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1484 }
1485}
1486
Jaewoong Jung39982342020-01-14 10:27:18 -08001487func TestAndroidTest_FixTestConfig(t *testing.T) {
1488 ctx, _ := testJava(t, `
1489 android_app {
1490 name: "foo",
1491 srcs: ["a.java"],
1492 package_name: "com.android.foo",
1493 sdk_version: "current",
1494 }
1495
1496 android_test {
1497 name: "foo_test",
1498 srcs: ["b.java"],
1499 instrumentation_for: "foo",
1500 }
1501
1502 android_test {
1503 name: "bar_test",
1504 srcs: ["b.java"],
1505 package_name: "com.android.bar.test",
1506 instrumentation_for: "foo",
1507 }
1508
1509 override_android_test {
1510 name: "baz_test",
1511 base: "foo_test",
1512 package_name: "com.android.baz.test",
1513 }
1514 `)
1515
1516 testCases := []struct {
1517 moduleName string
1518 variantName string
1519 expectedFlags []string
1520 }{
1521 {
1522 moduleName: "foo_test",
1523 variantName: "android_common",
1524 },
1525 {
1526 moduleName: "bar_test",
1527 variantName: "android_common",
1528 expectedFlags: []string{
1529 "--manifest " + buildDir + "/.intermediates/bar_test/android_common/manifest_fixer/AndroidManifest.xml",
1530 "--package-name com.android.bar.test",
1531 },
1532 },
1533 {
1534 moduleName: "foo_test",
1535 variantName: "android_common_baz_test",
1536 expectedFlags: []string{
1537 "--manifest " + buildDir +
1538 "/.intermediates/foo_test/android_common_baz_test/manifest_fixer/AndroidManifest.xml",
1539 "--package-name com.android.baz.test",
1540 "--test-file-name baz_test.apk",
1541 },
1542 },
1543 }
1544
1545 for _, test := range testCases {
1546 variant := ctx.ModuleForTests(test.moduleName, test.variantName)
1547 params := variant.MaybeOutput("test_config_fixer/AndroidTest.xml")
1548
1549 if len(test.expectedFlags) > 0 {
1550 if params.Rule == nil {
1551 t.Errorf("test_config_fixer was expected to run, but didn't")
1552 } else {
1553 for _, flag := range test.expectedFlags {
1554 if !strings.Contains(params.RuleParams.Command, flag) {
1555 t.Errorf("Flag %q was not found in command: %q", flag, params.RuleParams.Command)
1556 }
1557 }
1558 }
1559 } else {
1560 if params.Rule != nil {
1561 t.Errorf("test_config_fixer was not expected to run, but did: %q", params.RuleParams.Command)
1562 }
1563 }
1564
1565 }
1566}
1567
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001568func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001569 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001570 android_app_import {
1571 name: "foo",
1572 apk: "prebuilts/apk/app.apk",
1573 certificate: "platform",
1574 dex_preopt: {
1575 enabled: true,
1576 },
1577 }
1578 `)
1579
1580 variant := ctx.ModuleForTests("foo", "android_common")
1581
1582 // Check dexpreopt outputs.
1583 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1584 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1585 t.Errorf("can't find dexpreopt outputs")
1586 }
1587
1588 // Check cert signing flag.
1589 signedApk := variant.Output("signed/foo.apk")
1590 signingFlag := signedApk.Args["certificates"]
1591 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
1592 if expected != signingFlag {
1593 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1594 }
1595}
1596
1597func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001598 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001599 android_app_import {
1600 name: "foo",
1601 apk: "prebuilts/apk/app.apk",
1602 certificate: "platform",
1603 dex_preopt: {
1604 enabled: false,
1605 },
1606 }
1607 `)
1608
1609 variant := ctx.ModuleForTests("foo", "android_common")
1610
1611 // Check dexpreopt outputs. They shouldn't exist.
1612 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
1613 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
1614 t.Errorf("dexpreopt shouldn't have run.")
1615 }
1616}
1617
1618func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001619 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001620 android_app_import {
1621 name: "foo",
1622 apk: "prebuilts/apk/app.apk",
1623 presigned: true,
1624 dex_preopt: {
1625 enabled: true,
1626 },
1627 }
1628 `)
1629
1630 variant := ctx.ModuleForTests("foo", "android_common")
1631
1632 // Check dexpreopt outputs.
1633 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1634 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1635 t.Errorf("can't find dexpreopt outputs")
1636 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01001637 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001638 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
1639 t.Errorf("signing rule shouldn't be included.")
1640 }
1641 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
1642 t.Errorf("can't find aligning rule")
1643 }
1644}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001645
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001646func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
1647 ctx, _ := testJava(t, `
1648 android_app_import {
1649 name: "foo",
1650 apk: "prebuilts/apk/app.apk",
1651 default_dev_cert: true,
1652 dex_preopt: {
1653 enabled: true,
1654 },
1655 }
1656 `)
1657
1658 variant := ctx.ModuleForTests("foo", "android_common")
1659
1660 // Check dexpreopt outputs.
1661 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1662 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1663 t.Errorf("can't find dexpreopt outputs")
1664 }
1665
1666 // Check cert signing flag.
1667 signedApk := variant.Output("signed/foo.apk")
1668 signingFlag := signedApk.Args["certificates"]
1669 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
1670 if expected != signingFlag {
1671 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1672 }
1673}
1674
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001675func TestAndroidAppImport_DpiVariants(t *testing.T) {
1676 bp := `
1677 android_app_import {
1678 name: "foo",
1679 apk: "prebuilts/apk/app.apk",
1680 dpi_variants: {
1681 xhdpi: {
1682 apk: "prebuilts/apk/app_xhdpi.apk",
1683 },
1684 xxhdpi: {
1685 apk: "prebuilts/apk/app_xxhdpi.apk",
1686 },
1687 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001688 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001689 dex_preopt: {
1690 enabled: true,
1691 },
1692 }
1693 `
1694 testCases := []struct {
1695 name string
1696 aaptPreferredConfig *string
1697 aaptPrebuiltDPI []string
1698 expected string
1699 }{
1700 {
1701 name: "no preferred",
1702 aaptPreferredConfig: nil,
1703 aaptPrebuiltDPI: []string{},
1704 expected: "prebuilts/apk/app.apk",
1705 },
1706 {
1707 name: "AAPTPreferredConfig matches",
1708 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001709 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001710 expected: "prebuilts/apk/app_xhdpi.apk",
1711 },
1712 {
1713 name: "AAPTPrebuiltDPI matches",
1714 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1715 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
1716 expected: "prebuilts/apk/app_xxhdpi.apk",
1717 },
1718 {
1719 name: "non-first AAPTPrebuiltDPI matches",
1720 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1721 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
1722 expected: "prebuilts/apk/app_xhdpi.apk",
1723 },
1724 {
1725 name: "no matches",
1726 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1727 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
1728 expected: "prebuilts/apk/app.apk",
1729 },
1730 }
1731
1732 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
1733 for _, test := range testCases {
Colin Cross98be1bb2019-12-13 20:41:13 -08001734 config := testAppConfig(nil, bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001735 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
1736 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Colin Cross98be1bb2019-12-13 20:41:13 -08001737 ctx := testContext()
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001738
1739 run(t, ctx, config)
1740
1741 variant := ctx.ModuleForTests("foo", "android_common")
1742 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
1743 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
1744 if len(matches) != 2 {
1745 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
1746 }
1747 if test.expected != matches[1] {
1748 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
1749 }
1750 }
1751}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001752
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07001753func TestAndroidAppImport_Filename(t *testing.T) {
1754 ctx, config := testJava(t, `
1755 android_app_import {
1756 name: "foo",
1757 apk: "prebuilts/apk/app.apk",
1758 presigned: true,
1759 }
1760
1761 android_app_import {
1762 name: "bar",
1763 apk: "prebuilts/apk/app.apk",
1764 presigned: true,
1765 filename: "bar_sample.apk"
1766 }
1767 `)
1768
1769 testCases := []struct {
1770 name string
1771 expected string
1772 }{
1773 {
1774 name: "foo",
1775 expected: "foo.apk",
1776 },
1777 {
1778 name: "bar",
1779 expected: "bar_sample.apk",
1780 },
1781 }
1782
1783 for _, test := range testCases {
1784 variant := ctx.ModuleForTests(test.name, "android_common")
1785 if variant.MaybeOutput(test.expected).Rule == nil {
1786 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
1787 }
1788
1789 a := variant.Module().(*AndroidAppImport)
1790 expectedValues := []string{test.expected}
1791 actualValues := android.AndroidMkEntriesForTest(
Jiyong Park0b0e1b92019-12-03 13:24:29 +09001792 t, config, "", a)[0].EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07001793 if !reflect.DeepEqual(actualValues, expectedValues) {
1794 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
1795 actualValues, expectedValues)
1796 }
1797 }
1798}
1799
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001800func TestAndroidAppImport_ArchVariants(t *testing.T) {
1801 // The test config's target arch is ARM64.
1802 testCases := []struct {
1803 name string
1804 bp string
1805 expected string
1806 }{
1807 {
1808 name: "matching arch",
1809 bp: `
1810 android_app_import {
1811 name: "foo",
1812 apk: "prebuilts/apk/app.apk",
1813 arch: {
1814 arm64: {
1815 apk: "prebuilts/apk/app_arm64.apk",
1816 },
1817 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001818 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001819 dex_preopt: {
1820 enabled: true,
1821 },
1822 }
1823 `,
1824 expected: "prebuilts/apk/app_arm64.apk",
1825 },
1826 {
1827 name: "no matching arch",
1828 bp: `
1829 android_app_import {
1830 name: "foo",
1831 apk: "prebuilts/apk/app.apk",
1832 arch: {
1833 arm: {
1834 apk: "prebuilts/apk/app_arm.apk",
1835 },
1836 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001837 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001838 dex_preopt: {
1839 enabled: true,
1840 },
1841 }
1842 `,
1843 expected: "prebuilts/apk/app.apk",
1844 },
1845 }
1846
1847 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
1848 for _, test := range testCases {
1849 ctx, _ := testJava(t, test.bp)
1850
1851 variant := ctx.ModuleForTests("foo", "android_common")
1852 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
1853 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
1854 if len(matches) != 2 {
1855 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
1856 }
1857 if test.expected != matches[1] {
1858 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
1859 }
1860 }
1861}
1862
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001863func TestAndroidTestImport(t *testing.T) {
1864 ctx, config := testJava(t, `
1865 android_test_import {
1866 name: "foo",
1867 apk: "prebuilts/apk/app.apk",
1868 presigned: true,
1869 data: [
1870 "testdata/data",
1871 ],
1872 }
1873 `)
1874
1875 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
1876
1877 // Check android mks.
Jiyong Park0b0e1b92019-12-03 13:24:29 +09001878 entries := android.AndroidMkEntriesForTest(t, config, "", test)[0]
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001879 expected := []string{"tests"}
1880 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
1881 if !reflect.DeepEqual(expected, actual) {
1882 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
1883 }
1884 expected = []string{"testdata/data:testdata/data"}
1885 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
1886 if !reflect.DeepEqual(expected, actual) {
1887 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
1888 }
1889}
1890
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001891func TestAndroidTestImport_NoJinUncompressForPresigned(t *testing.T) {
1892 ctx, _ := testJava(t, `
1893 android_test_import {
1894 name: "foo",
1895 apk: "prebuilts/apk/app.apk",
1896 certificate: "cert/new_cert",
1897 data: [
1898 "testdata/data",
1899 ],
1900 }
1901
1902 android_test_import {
1903 name: "foo_presigned",
1904 apk: "prebuilts/apk/app.apk",
1905 presigned: true,
1906 data: [
1907 "testdata/data",
1908 ],
1909 }
1910 `)
1911
1912 variant := ctx.ModuleForTests("foo", "android_common")
1913 jniRule := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
1914 if !strings.HasPrefix(jniRule, "if (zipinfo") {
1915 t.Errorf("Unexpected JNI uncompress rule command: " + jniRule)
1916 }
1917
1918 variant = ctx.ModuleForTests("foo_presigned", "android_common")
1919 jniRule = variant.Output("jnis-uncompressed/foo_presigned.apk").BuildParams.Rule.String()
1920 if jniRule != android.Cp.String() {
1921 t.Errorf("Unexpected JNI uncompress rule: " + jniRule)
1922 }
1923}
1924
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001925func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001926 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001927 cc_library {
1928 name: "libjni",
Peter Collingbournead84f972019-12-17 16:46:18 -08001929 sdk_version: "current",
1930 stl: "c++_shared",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001931 }
1932
1933 android_test {
1934 name: "stl",
1935 jni_libs: ["libjni"],
1936 compile_multilib: "both",
1937 sdk_version: "current",
1938 stl: "c++_shared",
1939 }
1940
1941 android_test {
1942 name: "system",
1943 jni_libs: ["libjni"],
1944 compile_multilib: "both",
1945 sdk_version: "current",
1946 }
1947 `)
1948
1949 testCases := []struct {
1950 name string
1951 jnis []string
1952 }{
1953 {"stl",
1954 []string{
1955 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07001956 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001957 },
1958 },
1959 {"system",
1960 []string{
1961 "libjni.so",
1962 },
1963 },
1964 }
1965
1966 for _, test := range testCases {
1967 t.Run(test.name, func(t *testing.T) {
1968 app := ctx.ModuleForTests(test.name, "android_common")
1969 jniLibZip := app.Output("jnilibs.zip")
1970 var jnis []string
1971 args := strings.Fields(jniLibZip.Args["jarArgs"])
1972 for i := 0; i < len(args); i++ {
1973 if args[i] == "-f" {
1974 jnis = append(jnis, args[i+1])
1975 i += 1
1976 }
1977 }
1978 jnisJoined := strings.Join(jnis, " ")
1979 for _, jni := range test.jnis {
1980 if !strings.Contains(jnisJoined, jni) {
1981 t.Errorf("missing jni %q in %q", jni, jnis)
1982 }
1983 }
1984 })
1985 }
1986}
Colin Cross50ddcc42019-05-16 12:28:22 -07001987
1988func TestUsesLibraries(t *testing.T) {
1989 bp := `
1990 java_sdk_library {
1991 name: "foo",
1992 srcs: ["a.java"],
1993 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001994 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07001995 }
1996
1997 java_sdk_library {
1998 name: "bar",
1999 srcs: ["a.java"],
2000 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002001 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002002 }
2003
2004 android_app {
2005 name: "app",
2006 srcs: ["a.java"],
2007 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002008 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07002009 optional_uses_libs: [
2010 "bar",
2011 "baz",
2012 ],
2013 }
2014
2015 android_app_import {
2016 name: "prebuilt",
2017 apk: "prebuilts/apk/app.apk",
2018 certificate: "platform",
2019 uses_libs: ["foo"],
2020 optional_uses_libs: [
2021 "bar",
2022 "baz",
2023 ],
2024 }
2025 `
2026
Colin Cross98be1bb2019-12-13 20:41:13 -08002027 config := testAppConfig(nil, bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07002028 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
2029
Colin Cross98be1bb2019-12-13 20:41:13 -08002030 ctx := testContext()
Colin Cross50ddcc42019-05-16 12:28:22 -07002031
2032 run(t, ctx, config)
2033
2034 app := ctx.ModuleForTests("app", "android_common")
2035 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
2036
2037 // Test that all libraries are verified
2038 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
2039 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
2040 t.Errorf("wanted %q in %q", w, cmd)
2041 }
2042
2043 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
2044 t.Errorf("wanted %q in %q", w, cmd)
2045 }
2046
2047 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
2048
2049 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
2050 t.Errorf("wanted %q in %q", w, cmd)
2051 }
2052
2053 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
2054 t.Errorf("wanted %q in %q", w, cmd)
2055 }
2056
2057 // Test that only present libraries are preopted
2058 cmd = app.Rule("dexpreopt").RuleParams.Command
2059
2060 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2061 t.Errorf("wanted %q in %q", w, cmd)
2062 }
2063
2064 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
2065
2066 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
2067 t.Errorf("wanted %q in %q", w, cmd)
2068 }
2069}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002070
2071func TestCodelessApp(t *testing.T) {
2072 testCases := []struct {
2073 name string
2074 bp string
2075 noCode bool
2076 }{
2077 {
2078 name: "normal",
2079 bp: `
2080 android_app {
2081 name: "foo",
2082 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002083 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002084 }
2085 `,
2086 noCode: false,
2087 },
2088 {
2089 name: "app without sources",
2090 bp: `
2091 android_app {
2092 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002093 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002094 }
2095 `,
2096 noCode: true,
2097 },
2098 {
2099 name: "app with libraries",
2100 bp: `
2101 android_app {
2102 name: "foo",
2103 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002104 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002105 }
2106
2107 java_library {
2108 name: "lib",
2109 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002110 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002111 }
2112 `,
2113 noCode: false,
2114 },
2115 {
2116 name: "app with sourceless libraries",
2117 bp: `
2118 android_app {
2119 name: "foo",
2120 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002121 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002122 }
2123
2124 java_library {
2125 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002126 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07002127 }
2128 `,
2129 // TODO(jungjw): this should probably be true
2130 noCode: false,
2131 },
2132 }
2133
2134 for _, test := range testCases {
2135 t.Run(test.name, func(t *testing.T) {
2136 ctx := testApp(t, test.bp)
2137
2138 foo := ctx.ModuleForTests("foo", "android_common")
2139 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
2140 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
2141 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
2142 }
2143 })
2144 }
2145}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002146
2147func TestEmbedNotice(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07002148 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002149 android_app {
2150 name: "foo",
2151 srcs: ["a.java"],
2152 static_libs: ["javalib"],
2153 jni_libs: ["libjni"],
2154 notice: "APP_NOTICE",
2155 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002156 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002157 }
2158
2159 // No embed_notice flag
2160 android_app {
2161 name: "bar",
2162 srcs: ["a.java"],
2163 jni_libs: ["libjni"],
2164 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09002165 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002166 }
2167
2168 // No NOTICE files
2169 android_app {
2170 name: "baz",
2171 srcs: ["a.java"],
2172 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09002173 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002174 }
2175
2176 cc_library {
2177 name: "libjni",
2178 system_shared_libs: [],
2179 stl: "none",
2180 notice: "LIB_NOTICE",
Colin Cross094cde42020-02-15 10:38:00 -08002181 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002182 }
2183
2184 java_library {
2185 name: "javalib",
2186 srcs: [
2187 ":gen",
2188 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002189 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002190 }
2191
2192 genrule {
2193 name: "gen",
2194 tools: ["gentool"],
2195 out: ["gen.java"],
2196 notice: "GENRULE_NOTICE",
2197 }
2198
2199 java_binary_host {
2200 name: "gentool",
2201 srcs: ["b.java"],
2202 notice: "TOOL_NOTICE",
2203 }
2204 `)
2205
2206 // foo has NOTICE files to process, and embed_notices is true.
2207 foo := ctx.ModuleForTests("foo", "android_common")
2208 // verify merge notices rule.
2209 mergeNotices := foo.Rule("mergeNoticesRule")
2210 noticeInputs := mergeNotices.Inputs.Strings()
2211 // TOOL_NOTICE should be excluded as it's a host module.
2212 if len(mergeNotices.Inputs) != 3 {
2213 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
2214 }
2215 if !inList("APP_NOTICE", noticeInputs) {
2216 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
2217 }
2218 if !inList("LIB_NOTICE", noticeInputs) {
2219 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
2220 }
2221 if !inList("GENRULE_NOTICE", noticeInputs) {
2222 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
2223 }
2224 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
2225 res := foo.Output("package-res.apk")
2226 aapt2Flags := res.Args["flags"]
2227 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
2228 if !strings.Contains(aapt2Flags, e) {
2229 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
2230 }
2231
2232 // bar has NOTICE files to process, but embed_notices is not set.
2233 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002234 res = bar.Output("package-res.apk")
2235 aapt2Flags = res.Args["flags"]
2236 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
2237 if strings.Contains(aapt2Flags, e) {
2238 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002239 }
2240
2241 // baz's embed_notice is true, but it doesn't have any NOTICE files.
2242 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07002243 res = baz.Output("package-res.apk")
2244 aapt2Flags = res.Args["flags"]
2245 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
2246 if strings.Contains(aapt2Flags, e) {
2247 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07002248 }
2249}
Colin Cross53a87f52019-06-25 13:35:30 -07002250
2251func TestUncompressDex(t *testing.T) {
2252 testCases := []struct {
2253 name string
2254 bp string
2255
2256 uncompressedPlatform bool
2257 uncompressedUnbundled bool
2258 }{
2259 {
2260 name: "normal",
2261 bp: `
2262 android_app {
2263 name: "foo",
2264 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002265 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002266 }
2267 `,
2268 uncompressedPlatform: true,
2269 uncompressedUnbundled: false,
2270 },
2271 {
2272 name: "use_embedded_dex",
2273 bp: `
2274 android_app {
2275 name: "foo",
2276 use_embedded_dex: true,
2277 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002278 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002279 }
2280 `,
2281 uncompressedPlatform: true,
2282 uncompressedUnbundled: true,
2283 },
2284 {
2285 name: "privileged",
2286 bp: `
2287 android_app {
2288 name: "foo",
2289 privileged: true,
2290 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09002291 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07002292 }
2293 `,
2294 uncompressedPlatform: true,
2295 uncompressedUnbundled: true,
2296 },
2297 }
2298
2299 test := func(t *testing.T, bp string, want bool, unbundled bool) {
2300 t.Helper()
2301
Colin Cross98be1bb2019-12-13 20:41:13 -08002302 config := testAppConfig(nil, bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07002303 if unbundled {
2304 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
2305 }
2306
Colin Cross98be1bb2019-12-13 20:41:13 -08002307 ctx := testContext()
Colin Cross53a87f52019-06-25 13:35:30 -07002308
2309 run(t, ctx, config)
2310
2311 foo := ctx.ModuleForTests("foo", "android_common")
2312 dex := foo.Rule("r8")
2313 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
2314 aligned := foo.MaybeRule("zipalign").Rule != nil
2315
2316 if uncompressedInDexJar != want {
2317 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
2318 }
2319
2320 if aligned != want {
2321 t.Errorf("want aligned %v, got %v", want, aligned)
2322 }
2323 }
2324
2325 for _, tt := range testCases {
2326 t.Run(tt.name, func(t *testing.T) {
2327 t.Run("platform", func(t *testing.T) {
2328 test(t, tt.bp, tt.uncompressedPlatform, false)
2329 })
2330 t.Run("unbundled", func(t *testing.T) {
2331 test(t, tt.bp, tt.uncompressedUnbundled, true)
2332 })
2333 })
2334 }
2335}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07002336
2337func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
2338 if expectedValue != "" {
2339 expectedFlag := "--" + flagName + " " + expectedValue
2340 if !strings.Contains(aapt2Flags, expectedFlag) {
2341 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
2342 }
2343 } else {
2344 unexpectedFlag := "--" + flagName
2345 if strings.Contains(aapt2Flags, unexpectedFlag) {
2346 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
2347 }
2348 }
2349}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002350
2351func TestRuntimeResourceOverlay(t *testing.T) {
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002352 fs := map[string][]byte{
2353 "baz/res/res/values/strings.xml": nil,
2354 "bar/res/res/values/strings.xml": nil,
2355 }
2356 bp := `
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002357 runtime_resource_overlay {
2358 name: "foo",
2359 certificate: "platform",
2360 product_specific: true,
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002361 static_libs: ["bar"],
2362 resource_libs: ["baz"],
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002363 aaptflags: ["--keep-raw-values"],
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002364 }
2365
2366 runtime_resource_overlay {
2367 name: "foo_themed",
2368 certificate: "platform",
2369 product_specific: true,
2370 theme: "faza",
2371 }
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002372
2373 android_library {
2374 name: "bar",
2375 resource_dirs: ["bar/res"],
2376 }
2377
2378 android_app {
2379 name: "baz",
2380 sdk_version: "current",
2381 resource_dirs: ["baz/res"],
2382 }
2383 `
2384 config := testAppConfig(nil, bp, fs)
2385 ctx := testContext()
2386 run(t, ctx, config)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002387
2388 m := ctx.ModuleForTests("foo", "android_common")
2389
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08002390 // Check AAPT2 link flags.
2391 aapt2Flags := m.Output("package-res.apk").Args["flags"]
2392 expectedFlags := []string{"--keep-raw-values", "--no-resource-deduping", "--no-resource-removal"}
2393 absentFlags := android.RemoveListFromList(expectedFlags, strings.Split(aapt2Flags, " "))
2394 if len(absentFlags) > 0 {
2395 t.Errorf("expected values, %q are missing in aapt2 link flags, %q", absentFlags, aapt2Flags)
2396 }
2397
Jaewoong Jungfe3c7f62020-04-09 16:15:30 -07002398 // Check overlay.list output for static_libs dependency.
2399 overlayList := m.Output("aapt2/overlay.list").Inputs.Strings()
2400 staticLibPackage := buildDir + "/.intermediates/bar/android_common/package-res.apk"
2401 if !inList(staticLibPackage, overlayList) {
2402 t.Errorf("Stactic lib res package %q missing in overlay list: %q", staticLibPackage, overlayList)
2403 }
2404
2405 // Check AAPT2 link flags for resource_libs dependency.
2406 resourceLibFlag := "-I " + buildDir + "/.intermediates/baz/android_common/package-res.apk"
2407 if !strings.Contains(aapt2Flags, resourceLibFlag) {
2408 t.Errorf("Resource lib flag %q missing in aapt2 link flags: %q", resourceLibFlag, aapt2Flags)
2409 }
2410
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002411 // Check cert signing flag.
2412 signedApk := m.Output("signed/foo.apk")
2413 signingFlag := signedApk.Args["certificates"]
2414 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
2415 if expected != signingFlag {
2416 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
2417 }
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002418 path := android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_CERTIFICATE"]
2419 expectedPath := []string{"build/make/target/product/security/platform.x509.pem"}
2420 if !reflect.DeepEqual(path, expectedPath) {
2421 t.Errorf("Unexpected LOCAL_CERTIFICATE value: %v, expected: %v", path, expectedPath)
2422 }
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002423
2424 // Check device location.
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08002425 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2426 expectedPath = []string{"/tmp/target/product/test_device/product/overlay"}
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08002427 if !reflect.DeepEqual(path, expectedPath) {
2428 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2429 }
2430
2431 // A themed module has a different device location
2432 m = ctx.ModuleForTests("foo_themed", "android_common")
2433 path = android.AndroidMkEntriesForTest(t, config, "", m.Module())[0].EntryMap["LOCAL_MODULE_PATH"]
2434 expectedPath = []string{"/tmp/target/product/test_device/product/overlay/faza"}
2435 if !reflect.DeepEqual(path, expectedPath) {
2436 t.Errorf("Unexpected LOCAL_MODULE_PATH value: %v, expected: %v", path, expectedPath)
2437 }
2438}