blob: fc8cf8e90430917ebf4d5fd4407e2605007d83e8 [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
Jaewoong Jungf9a04432019-07-17 11:15:09 -070046func testAppContext(bp string, fs map[string][]byte) *android.TestContext {
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
Jaewoong Jungf9a04432019-07-17 11:15:09 -070056 return testContext(bp, appFS)
Colin Cross527012a2017-11-30 22:56:16 -080057}
58
59func testApp(t *testing.T, bp string) *android.TestContext {
60 config := testConfig(nil)
61
Jaewoong Jungf9a04432019-07-17 11:15:09 -070062 ctx := testAppContext(bp, nil)
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
Colin Cross0ddae7f2019-02-07 15:30:01 -0800178func TestResourceDirs(t *testing.T) {
179 testCases := []struct {
180 name string
181 prop string
182 resources []string
183 }{
184 {
185 name: "no resource_dirs",
186 prop: "",
187 resources: []string{"res/res/values/strings.xml"},
188 },
189 {
190 name: "resource_dirs",
191 prop: `resource_dirs: ["res"]`,
192 resources: []string{"res/res/values/strings.xml"},
193 },
194 {
195 name: "empty resource_dirs",
196 prop: `resource_dirs: []`,
197 resources: nil,
198 },
199 }
200
201 fs := map[string][]byte{
202 "res/res/values/strings.xml": nil,
203 }
204
205 bp := `
206 android_app {
207 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900208 sdk_version: "current",
Colin Cross0ddae7f2019-02-07 15:30:01 -0800209 %s
210 }
211 `
212
213 for _, testCase := range testCases {
214 t.Run(testCase.name, func(t *testing.T) {
215 config := testConfig(nil)
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700216 ctx := testContext(fmt.Sprintf(bp, testCase.prop), fs)
Colin Cross0ddae7f2019-02-07 15:30:01 -0800217 run(t, ctx, config)
218
219 module := ctx.ModuleForTests("foo", "android_common")
220 resourceList := module.MaybeOutput("aapt2/res.list")
221
222 var resources []string
223 if resourceList.Rule != nil {
224 for _, compiledResource := range resourceList.Inputs.Strings() {
225 resources = append(resources, module.Output(compiledResource).Inputs.Strings()...)
226 }
227 }
228
229 if !reflect.DeepEqual(resources, testCase.resources) {
230 t.Errorf("expected resource files %q, got %q",
231 testCase.resources, resources)
232 }
233 })
234 }
235}
236
Colin Crossbec85302019-02-13 13:15:46 -0800237func TestAndroidResources(t *testing.T) {
Colin Cross5c4791c2019-02-01 11:44:44 -0800238 testCases := []struct {
239 name string
240 enforceRROTargets []string
241 enforceRROExcludedOverlays []string
Colin Crossbec85302019-02-13 13:15:46 -0800242 resourceFiles map[string][]string
Colin Cross5c4791c2019-02-01 11:44:44 -0800243 overlayFiles map[string][]string
244 rroDirs map[string][]string
245 }{
246 {
247 name: "no RRO",
248 enforceRROTargets: nil,
249 enforceRROExcludedOverlays: nil,
Colin Crossbec85302019-02-13 13:15:46 -0800250 resourceFiles: map[string][]string{
251 "foo": nil,
252 "bar": {"bar/res/res/values/strings.xml"},
253 "lib": nil,
254 "lib2": {"lib2/res/res/values/strings.xml"},
255 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800256 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800257 "foo": {
258 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800259 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000260 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800261 "foo/res/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800262 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
263 "device/vendor/blah/overlay/foo/res/values/strings.xml",
Anton Hansson53c88442019-03-18 15:53:16 +0000264 "product/vendor/blah/overlay/foo/res/values/strings.xml",
Colin Cross5c4791c2019-02-01 11:44:44 -0800265 },
Colin Crossbec85302019-02-13 13:15:46 -0800266 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800267 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
268 "device/vendor/blah/overlay/bar/res/values/strings.xml",
269 },
Colin Crossbec85302019-02-13 13:15:46 -0800270 "lib": {
271 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
272 "lib/res/res/values/strings.xml",
273 "device/vendor/blah/overlay/lib/res/values/strings.xml",
274 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800275 },
276 rroDirs: map[string][]string{
277 "foo": nil,
278 "bar": nil,
279 },
280 },
281 {
282 name: "enforce RRO on foo",
283 enforceRROTargets: []string{"foo"},
284 enforceRROExcludedOverlays: []string{"device/vendor/blah/static_overlay"},
Colin Crossbec85302019-02-13 13:15:46 -0800285 resourceFiles: map[string][]string{
286 "foo": nil,
287 "bar": {"bar/res/res/values/strings.xml"},
288 "lib": nil,
289 "lib2": {"lib2/res/res/values/strings.xml"},
290 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800291 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800292 "foo": {
293 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800294 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000295 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800296 "foo/res/res/values/strings.xml",
297 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
298 },
Colin Crossbec85302019-02-13 13:15:46 -0800299 "bar": {
Colin Cross5c4791c2019-02-01 11:44:44 -0800300 "device/vendor/blah/static_overlay/bar/res/values/strings.xml",
301 "device/vendor/blah/overlay/bar/res/values/strings.xml",
302 },
Colin Crossbec85302019-02-13 13:15:46 -0800303 "lib": {
304 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
305 "lib/res/res/values/strings.xml",
306 "device/vendor/blah/overlay/lib/res/values/strings.xml",
307 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800308 },
Colin Crossc1c37552019-01-31 11:42:41 -0800309
Colin Cross5c4791c2019-02-01 11:44:44 -0800310 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800311 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000312 "device:device/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800313 // Enforce RRO on "foo" could imply RRO on static dependencies, but for now it doesn't.
314 // "device/vendor/blah/overlay/lib/res",
Anton Hansson53c88442019-03-18 15:53:16 +0000315 "product:product/vendor/blah/overlay/foo/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800316 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800317 "bar": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800318 "lib": nil,
Colin Cross5c4791c2019-02-01 11:44:44 -0800319 },
320 },
321 {
322 name: "enforce RRO on all",
323 enforceRROTargets: []string{"*"},
324 enforceRROExcludedOverlays: []string{
325 // Excluding specific apps/res directories also allowed.
326 "device/vendor/blah/static_overlay/foo",
327 "device/vendor/blah/static_overlay/bar/res",
328 },
Colin Crossbec85302019-02-13 13:15:46 -0800329 resourceFiles: map[string][]string{
330 "foo": nil,
331 "bar": {"bar/res/res/values/strings.xml"},
332 "lib": nil,
333 "lib2": {"lib2/res/res/values/strings.xml"},
334 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800335 overlayFiles: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800336 "foo": {
337 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800338 buildDir + "/.intermediates/lib/android_common/package-res.apk",
Anton Hansson53c88442019-03-18 15:53:16 +0000339 buildDir + "/.intermediates/lib3/android_common/package-res.apk",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800340 "foo/res/res/values/strings.xml",
341 "device/vendor/blah/static_overlay/foo/res/values/strings.xml",
342 },
Colin Crossbec85302019-02-13 13:15:46 -0800343 "bar": {"device/vendor/blah/static_overlay/bar/res/values/strings.xml"},
344 "lib": {
345 buildDir + "/.intermediates/lib2/android_common/package-res.apk",
346 "lib/res/res/values/strings.xml",
347 },
Colin Cross5c4791c2019-02-01 11:44:44 -0800348 },
349 rroDirs: map[string][]string{
Colin Crossbec85302019-02-13 13:15:46 -0800350 "foo": {
Anton Hansson53c88442019-03-18 15:53:16 +0000351 "device:device/vendor/blah/overlay/foo/res",
352 "product:product/vendor/blah/overlay/foo/res",
353 // Lib dep comes after the direct deps
354 "device:device/vendor/blah/overlay/lib/res",
Colin Crossc1c37552019-01-31 11:42:41 -0800355 },
Anton Hansson53c88442019-03-18 15:53:16 +0000356 "bar": {"device:device/vendor/blah/overlay/bar/res"},
357 "lib": {"device:device/vendor/blah/overlay/lib/res"},
Colin Cross5c4791c2019-02-01 11:44:44 -0800358 },
359 },
360 }
361
Anton Hansson53c88442019-03-18 15:53:16 +0000362 deviceResourceOverlays := []string{
Colin Cross890ff552017-11-30 20:13:19 -0800363 "device/vendor/blah/overlay",
364 "device/vendor/blah/overlay2",
365 "device/vendor/blah/static_overlay",
366 }
367
Anton Hansson53c88442019-03-18 15:53:16 +0000368 productResourceOverlays := []string{
369 "product/vendor/blah/overlay",
370 }
371
Colin Cross890ff552017-11-30 20:13:19 -0800372 fs := map[string][]byte{
373 "foo/res/res/values/strings.xml": nil,
374 "bar/res/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800375 "lib/res/res/values/strings.xml": nil,
Colin Crossbec85302019-02-13 13:15:46 -0800376 "lib2/res/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800377 "device/vendor/blah/overlay/foo/res/values/strings.xml": nil,
378 "device/vendor/blah/overlay/bar/res/values/strings.xml": nil,
Colin Cross6ed7dea2019-01-31 14:44:30 -0800379 "device/vendor/blah/overlay/lib/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800380 "device/vendor/blah/static_overlay/foo/res/values/strings.xml": nil,
381 "device/vendor/blah/static_overlay/bar/res/values/strings.xml": nil,
382 "device/vendor/blah/overlay2/res/values/strings.xml": nil,
Anton Hansson53c88442019-03-18 15:53:16 +0000383 "product/vendor/blah/overlay/foo/res/values/strings.xml": nil,
Colin Cross890ff552017-11-30 20:13:19 -0800384 }
385
386 bp := `
387 android_app {
388 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900389 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800390 resource_dirs: ["foo/res"],
Anton Hansson53c88442019-03-18 15:53:16 +0000391 static_libs: ["lib", "lib3"],
Colin Cross890ff552017-11-30 20:13:19 -0800392 }
393
394 android_app {
395 name: "bar",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900396 sdk_version: "current",
Colin Cross890ff552017-11-30 20:13:19 -0800397 resource_dirs: ["bar/res"],
398 }
Colin Cross6ed7dea2019-01-31 14:44:30 -0800399
400 android_library {
401 name: "lib",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900402 sdk_version: "current",
Colin Cross6ed7dea2019-01-31 14:44:30 -0800403 resource_dirs: ["lib/res"],
Colin Crossbec85302019-02-13 13:15:46 -0800404 static_libs: ["lib2"],
405 }
406
407 android_library {
408 name: "lib2",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900409 sdk_version: "current",
Colin Crossbec85302019-02-13 13:15:46 -0800410 resource_dirs: ["lib2/res"],
Colin Cross6ed7dea2019-01-31 14:44:30 -0800411 }
Anton Hansson53c88442019-03-18 15:53:16 +0000412
413 // This library has the same resources as lib (should not lead to dupe RROs)
414 android_library {
415 name: "lib3",
Jeongik Cha75b83b02019-11-01 15:28:00 +0900416 sdk_version: "current",
Anton Hansson53c88442019-03-18 15:53:16 +0000417 resource_dirs: ["lib/res"]
418 }
Colin Cross890ff552017-11-30 20:13:19 -0800419 `
420
Colin Cross5c4791c2019-02-01 11:44:44 -0800421 for _, testCase := range testCases {
Colin Cross890ff552017-11-30 20:13:19 -0800422 t.Run(testCase.name, func(t *testing.T) {
423 config := testConfig(nil)
Anton Hansson53c88442019-03-18 15:53:16 +0000424 config.TestProductVariables.DeviceResourceOverlays = deviceResourceOverlays
425 config.TestProductVariables.ProductResourceOverlays = productResourceOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800426 if testCase.enforceRROTargets != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800427 config.TestProductVariables.EnforceRROTargets = testCase.enforceRROTargets
Colin Cross890ff552017-11-30 20:13:19 -0800428 }
429 if testCase.enforceRROExcludedOverlays != nil {
Colin Crossa74ca042019-01-31 14:31:51 -0800430 config.TestProductVariables.EnforceRROExcludedOverlays = testCase.enforceRROExcludedOverlays
Colin Cross890ff552017-11-30 20:13:19 -0800431 }
432
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700433 ctx := testAppContext(bp, fs)
Colin Cross890ff552017-11-30 20:13:19 -0800434 run(t, ctx, config)
435
Colin Crossbec85302019-02-13 13:15:46 -0800436 resourceListToFiles := func(module android.TestingModule, list []string) (files []string) {
437 for _, o := range list {
438 res := module.MaybeOutput(o)
439 if res.Rule != nil {
440 // If the overlay is compiled as part of this module (i.e. a .arsc.flat file),
441 // verify the inputs to the .arsc.flat rule.
442 files = append(files, res.Inputs.Strings()...)
443 } else {
444 // Otherwise, verify the full path to the output of the other module
445 files = append(files, o)
Anton Hansson94c93f32019-01-30 16:03:37 +0000446 }
Colin Cross890ff552017-11-30 20:13:19 -0800447 }
Colin Crossbec85302019-02-13 13:15:46 -0800448 return files
Colin Cross890ff552017-11-30 20:13:19 -0800449 }
450
Colin Crossbec85302019-02-13 13:15:46 -0800451 getResources := func(moduleName string) (resourceFiles, overlayFiles, rroDirs []string) {
452 module := ctx.ModuleForTests(moduleName, "android_common")
453 resourceList := module.MaybeOutput("aapt2/res.list")
454 if resourceList.Rule != nil {
455 resourceFiles = resourceListToFiles(module, resourceList.Inputs.Strings())
Anton Hansson0375a4f2019-01-24 14:39:19 +0000456 }
Colin Crossbec85302019-02-13 13:15:46 -0800457 overlayList := module.MaybeOutput("aapt2/overlay.list")
458 if overlayList.Rule != nil {
459 overlayFiles = resourceListToFiles(module, overlayList.Inputs.Strings())
460 }
461
Anton Hansson53c88442019-03-18 15:53:16 +0000462 for _, d := range module.Module().(AndroidLibraryDependency).ExportedRRODirs() {
463 var prefix string
464 if d.overlayType == device {
465 prefix = "device:"
466 } else if d.overlayType == product {
467 prefix = "product:"
468 } else {
469 t.Fatalf("Unexpected overlayType %d", d.overlayType)
470 }
471 rroDirs = append(rroDirs, prefix+d.path.String())
472 }
Colin Crossbec85302019-02-13 13:15:46 -0800473
474 return resourceFiles, overlayFiles, rroDirs
475 }
476
477 modules := []string{"foo", "bar", "lib", "lib2"}
478 for _, module := range modules {
479 resourceFiles, overlayFiles, rroDirs := getResources(module)
480
481 if !reflect.DeepEqual(resourceFiles, testCase.resourceFiles[module]) {
482 t.Errorf("expected %s resource files:\n %#v\n got:\n %#v",
483 module, testCase.resourceFiles[module], resourceFiles)
484 }
485 if !reflect.DeepEqual(overlayFiles, testCase.overlayFiles[module]) {
486 t.Errorf("expected %s overlay files:\n %#v\n got:\n %#v",
487 module, testCase.overlayFiles[module], overlayFiles)
488 }
489 if !reflect.DeepEqual(rroDirs, testCase.rroDirs[module]) {
Anton Hansson0375a4f2019-01-24 14:39:19 +0000490 t.Errorf("expected %s rroDirs: %#v\n got:\n %#v",
Colin Crossbec85302019-02-13 13:15:46 -0800491 module, testCase.rroDirs[module], rroDirs)
Anton Hansson0375a4f2019-01-24 14:39:19 +0000492 }
Colin Cross890ff552017-11-30 20:13:19 -0800493 }
Colin Cross890ff552017-11-30 20:13:19 -0800494 })
495 }
496}
Colin Crossd09b0b62018-04-18 11:06:47 -0700497
498func TestAppSdkVersion(t *testing.T) {
499 testCases := []struct {
500 name string
501 sdkVersion string
502 platformSdkInt int
503 platformSdkCodename string
504 platformSdkFinal bool
505 expectedMinSdkVersion string
Jeongik Cha538c0d02019-07-11 15:54:27 +0900506 platformApis bool
Colin Crossd09b0b62018-04-18 11:06:47 -0700507 }{
508 {
509 name: "current final SDK",
510 sdkVersion: "current",
511 platformSdkInt: 27,
512 platformSdkCodename: "REL",
513 platformSdkFinal: true,
514 expectedMinSdkVersion: "27",
515 },
516 {
517 name: "current non-final SDK",
518 sdkVersion: "current",
519 platformSdkInt: 27,
520 platformSdkCodename: "OMR1",
521 platformSdkFinal: false,
522 expectedMinSdkVersion: "OMR1",
523 },
524 {
525 name: "default final SDK",
526 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900527 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700528 platformSdkInt: 27,
529 platformSdkCodename: "REL",
530 platformSdkFinal: true,
531 expectedMinSdkVersion: "27",
532 },
533 {
534 name: "default non-final SDK",
535 sdkVersion: "",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900536 platformApis: true,
Colin Crossd09b0b62018-04-18 11:06:47 -0700537 platformSdkInt: 27,
538 platformSdkCodename: "OMR1",
539 platformSdkFinal: false,
540 expectedMinSdkVersion: "OMR1",
541 },
542 {
543 name: "14",
544 sdkVersion: "14",
545 expectedMinSdkVersion: "14",
546 },
547 }
548
549 for _, moduleType := range []string{"android_app", "android_library"} {
550 for _, test := range testCases {
551 t.Run(moduleType+" "+test.name, func(t *testing.T) {
Jeongik Cha538c0d02019-07-11 15:54:27 +0900552 platformApiProp := ""
553 if test.platformApis {
554 platformApiProp = "platform_apis: true,"
555 }
Colin Crossd09b0b62018-04-18 11:06:47 -0700556 bp := fmt.Sprintf(`%s {
557 name: "foo",
558 srcs: ["a.java"],
559 sdk_version: "%s",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900560 %s
561 }`, moduleType, test.sdkVersion, platformApiProp)
Colin Crossd09b0b62018-04-18 11:06:47 -0700562
563 config := testConfig(nil)
564 config.TestProductVariables.Platform_sdk_version = &test.platformSdkInt
565 config.TestProductVariables.Platform_sdk_codename = &test.platformSdkCodename
566 config.TestProductVariables.Platform_sdk_final = &test.platformSdkFinal
567
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700568 ctx := testAppContext(bp, nil)
Colin Crossd09b0b62018-04-18 11:06:47 -0700569
570 run(t, ctx, config)
571
572 foo := ctx.ModuleForTests("foo", "android_common")
573 link := foo.Output("package-res.apk")
574 linkFlags := strings.Split(link.Args["flags"], " ")
575 min := android.IndexList("--min-sdk-version", linkFlags)
576 target := android.IndexList("--target-sdk-version", linkFlags)
577
578 if min == -1 || target == -1 || min == len(linkFlags)-1 || target == len(linkFlags)-1 {
579 t.Fatalf("missing --min-sdk-version or --target-sdk-version in link flags: %q", linkFlags)
580 }
581
582 gotMinSdkVersion := linkFlags[min+1]
583 gotTargetSdkVersion := linkFlags[target+1]
584
585 if gotMinSdkVersion != test.expectedMinSdkVersion {
586 t.Errorf("incorrect --min-sdk-version, expected %q got %q",
587 test.expectedMinSdkVersion, gotMinSdkVersion)
588 }
589
590 if gotTargetSdkVersion != test.expectedMinSdkVersion {
591 t.Errorf("incorrect --target-sdk-version, expected %q got %q",
592 test.expectedMinSdkVersion, gotTargetSdkVersion)
593 }
594 })
595 }
596 }
597}
Colin Crossa4f08812018-10-02 22:03:40 -0700598
Paul Duffin50c217c2019-06-12 13:25:22 +0100599func TestJNIABI(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700600 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +0100601 cc_library {
602 name: "libjni",
603 system_shared_libs: [],
604 stl: "none",
605 }
606
607 android_test {
608 name: "test",
609 sdk_version: "core_platform",
610 jni_libs: ["libjni"],
611 }
612
613 android_test {
614 name: "test_first",
615 sdk_version: "core_platform",
616 compile_multilib: "first",
617 jni_libs: ["libjni"],
618 }
619
620 android_test {
621 name: "test_both",
622 sdk_version: "core_platform",
623 compile_multilib: "both",
624 jni_libs: ["libjni"],
625 }
626
627 android_test {
628 name: "test_32",
629 sdk_version: "core_platform",
630 compile_multilib: "32",
631 jni_libs: ["libjni"],
632 }
633
634 android_test {
635 name: "test_64",
636 sdk_version: "core_platform",
637 compile_multilib: "64",
638 jni_libs: ["libjni"],
639 }
640 `)
641
642 testCases := []struct {
643 name string
644 abis []string
645 }{
646 {"test", []string{"arm64-v8a"}},
647 {"test_first", []string{"arm64-v8a"}},
648 {"test_both", []string{"arm64-v8a", "armeabi-v7a"}},
649 {"test_32", []string{"armeabi-v7a"}},
650 {"test_64", []string{"arm64-v8a"}},
651 }
652
653 for _, test := range testCases {
654 t.Run(test.name, func(t *testing.T) {
655 app := ctx.ModuleForTests(test.name, "android_common")
656 jniLibZip := app.Output("jnilibs.zip")
657 var abis []string
658 args := strings.Fields(jniLibZip.Args["jarArgs"])
659 for i := 0; i < len(args); i++ {
660 if args[i] == "-P" {
661 abis = append(abis, filepath.Base(args[i+1]))
662 i++
663 }
664 }
665 if !reflect.DeepEqual(abis, test.abis) {
666 t.Errorf("want abis %v, got %v", test.abis, abis)
667 }
668 })
669 }
670}
671
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900672func TestAppSdkVersionByPartition(t *testing.T) {
673 testJavaError(t, "sdk_version must have a value when the module is located at vendor or product", `
674 android_app {
675 name: "foo",
676 srcs: ["a.java"],
677 vendor: true,
678 platform_apis: true,
679 }
680 `)
681
682 testJava(t, `
683 android_app {
684 name: "bar",
685 srcs: ["b.java"],
686 platform_apis: true,
687 }
688 `)
689
690 for _, enforce := range []bool{true, false} {
691
692 config := testConfig(nil)
693 config.TestProductVariables.EnforceProductPartitionInterface = proptools.BoolPtr(enforce)
694 bp := `
695 android_app {
696 name: "foo",
697 srcs: ["a.java"],
698 product_specific: true,
699 platform_apis: true,
700 }
701 `
702 if enforce {
703 testJavaErrorWithConfig(t, "sdk_version must have a value when the module is located at vendor or product", bp, config)
704 } else {
705 testJavaWithConfig(t, bp, config)
706 }
707 }
708}
709
Paul Duffin50c217c2019-06-12 13:25:22 +0100710func TestJNIPackaging(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700711 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Paul Duffin50c217c2019-06-12 13:25:22 +0100712 cc_library {
713 name: "libjni",
714 system_shared_libs: [],
715 stl: "none",
716 }
717
718 android_app {
719 name: "app",
720 jni_libs: ["libjni"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900721 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100722 }
723
724 android_app {
725 name: "app_noembed",
726 jni_libs: ["libjni"],
727 use_embedded_native_libs: false,
Jeongik Cha538c0d02019-07-11 15:54:27 +0900728 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100729 }
730
731 android_app {
732 name: "app_embed",
733 jni_libs: ["libjni"],
734 use_embedded_native_libs: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +0900735 sdk_version: "current",
Paul Duffin50c217c2019-06-12 13:25:22 +0100736 }
737
738 android_test {
739 name: "test",
740 sdk_version: "core_platform",
741 jni_libs: ["libjni"],
742 }
743
744 android_test {
745 name: "test_noembed",
746 sdk_version: "core_platform",
747 jni_libs: ["libjni"],
748 use_embedded_native_libs: false,
749 }
750
751 android_test_helper_app {
752 name: "test_helper",
753 sdk_version: "core_platform",
754 jni_libs: ["libjni"],
755 }
756
757 android_test_helper_app {
758 name: "test_helper_noembed",
759 sdk_version: "core_platform",
760 jni_libs: ["libjni"],
761 use_embedded_native_libs: false,
762 }
763 `)
764
765 testCases := []struct {
766 name string
767 packaged bool
768 compressed bool
769 }{
770 {"app", false, false},
771 {"app_noembed", false, false},
772 {"app_embed", true, false},
773 {"test", true, false},
774 {"test_noembed", true, true},
775 {"test_helper", true, false},
776 {"test_helper_noembed", true, true},
777 }
778
779 for _, test := range testCases {
780 t.Run(test.name, func(t *testing.T) {
781 app := ctx.ModuleForTests(test.name, "android_common")
782 jniLibZip := app.MaybeOutput("jnilibs.zip")
783 if g, w := (jniLibZip.Rule != nil), test.packaged; g != w {
784 t.Errorf("expected jni packaged %v, got %v", w, g)
785 }
786
787 if jniLibZip.Rule != nil {
788 if g, w := !strings.Contains(jniLibZip.Args["jarArgs"], "-L 0"), test.compressed; g != w {
789 t.Errorf("expected jni compressed %v, got %v", w, g)
790 }
791 }
792 })
793 }
Colin Cross47fa9d32019-03-26 10:51:39 -0700794}
795
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800796func TestCertificates(t *testing.T) {
797 testCases := []struct {
798 name string
799 bp string
800 certificateOverride string
801 expected string
802 }{
803 {
804 name: "default",
805 bp: `
806 android_app {
807 name: "foo",
808 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900809 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800810 }
811 `,
812 certificateOverride: "",
Dan Willemsen412160e2019-04-09 21:36:26 -0700813 expected: "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800814 },
815 {
816 name: "module certificate property",
817 bp: `
818 android_app {
819 name: "foo",
820 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900821 certificate: ":new_certificate",
822 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800823 }
824
825 android_app_certificate {
826 name: "new_certificate",
827 certificate: "cert/new_cert",
828 }
829 `,
830 certificateOverride: "",
831 expected: "cert/new_cert.x509.pem cert/new_cert.pk8",
832 },
833 {
834 name: "path certificate property",
835 bp: `
836 android_app {
837 name: "foo",
838 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900839 certificate: "expiredkey",
840 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800841 }
842 `,
843 certificateOverride: "",
Dan Willemsen412160e2019-04-09 21:36:26 -0700844 expected: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800845 },
846 {
847 name: "certificate overrides",
848 bp: `
849 android_app {
850 name: "foo",
851 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900852 certificate: "expiredkey",
853 sdk_version: "current",
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800854 }
855
856 android_app_certificate {
857 name: "new_certificate",
858 certificate: "cert/new_cert",
859 }
860 `,
861 certificateOverride: "foo:new_certificate",
862 expected: "cert/new_cert.x509.pem cert/new_cert.pk8",
863 },
864 }
865
866 for _, test := range testCases {
867 t.Run(test.name, func(t *testing.T) {
868 config := testConfig(nil)
869 if test.certificateOverride != "" {
870 config.TestProductVariables.CertificateOverrides = []string{test.certificateOverride}
871 }
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700872 ctx := testAppContext(test.bp, nil)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800873
874 run(t, ctx, config)
875 foo := ctx.ModuleForTests("foo", "android_common")
876
877 signapk := foo.Output("foo.apk")
878 signFlags := signapk.Args["certificates"]
879 if test.expected != signFlags {
880 t.Errorf("Incorrect signing flags, expected: %q, got: %q", test.expected, signFlags)
881 }
882 })
883 }
884}
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800885
886func TestPackageNameOverride(t *testing.T) {
887 testCases := []struct {
888 name string
889 bp string
890 packageNameOverride string
891 expected []string
892 }{
893 {
894 name: "default",
895 bp: `
896 android_app {
897 name: "foo",
898 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900899 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800900 }
901 `,
902 packageNameOverride: "",
903 expected: []string{
904 buildDir + "/.intermediates/foo/android_common/foo.apk",
905 buildDir + "/target/product/test_device/system/app/foo/foo.apk",
906 },
907 },
908 {
909 name: "overridden",
910 bp: `
911 android_app {
912 name: "foo",
913 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900914 sdk_version: "current",
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800915 }
916 `,
917 packageNameOverride: "foo:bar",
918 expected: []string{
919 // The package apk should be still be the original name for test dependencies.
Jaewoong Jung5a498812019-11-07 14:14:38 -0800920 buildDir + "/.intermediates/foo/android_common/bar.apk",
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800921 buildDir + "/target/product/test_device/system/app/bar/bar.apk",
922 },
923 },
924 }
925
926 for _, test := range testCases {
927 t.Run(test.name, func(t *testing.T) {
928 config := testConfig(nil)
929 if test.packageNameOverride != "" {
930 config.TestProductVariables.PackageNameOverrides = []string{test.packageNameOverride}
931 }
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700932 ctx := testAppContext(test.bp, nil)
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800933
934 run(t, ctx, config)
935 foo := ctx.ModuleForTests("foo", "android_common")
936
937 outputs := foo.AllOutputs()
938 outputMap := make(map[string]bool)
939 for _, o := range outputs {
940 outputMap[o] = true
941 }
942 for _, e := range test.expected {
943 if _, exist := outputMap[e]; !exist {
944 t.Errorf("Can't find %q in output files.\nAll outputs:%v", e, outputs)
945 }
946 }
947 })
948 }
949}
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800950
951func TestInstrumentationTargetOverridden(t *testing.T) {
952 bp := `
953 android_app {
954 name: "foo",
955 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900956 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800957 }
958
959 android_test {
960 name: "bar",
961 instrumentation_for: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +0900962 sdk_version: "current",
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800963 }
964 `
965 config := testConfig(nil)
966 config.TestProductVariables.ManifestPackageNameOverrides = []string{"foo:org.dandroid.bp"}
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700967 ctx := testAppContext(bp, nil)
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800968
969 run(t, ctx, config)
970
971 bar := ctx.ModuleForTests("bar", "android_common")
972 res := bar.Output("package-res.apk")
973 aapt2Flags := res.Args["flags"]
974 e := "--rename-instrumentation-target-package org.dandroid.bp"
975 if !strings.Contains(aapt2Flags, e) {
976 t.Errorf("target package renaming flag, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
977 }
978}
Jaewoong Jung525443a2019-02-28 15:35:54 -0800979
980func TestOverrideAndroidApp(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -0700981 ctx, _ := testJava(t, `
Jaewoong Jung525443a2019-02-28 15:35:54 -0800982 android_app {
983 name: "foo",
984 srcs: ["a.java"],
Jaewoong Junga641ee92019-03-27 11:17:14 -0700985 certificate: "expiredkey",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -0700986 overrides: ["qux"],
Jeongik Cha538c0d02019-07-11 15:54:27 +0900987 sdk_version: "current",
Jaewoong Jung525443a2019-02-28 15:35:54 -0800988 }
989
990 override_android_app {
991 name: "bar",
992 base: "foo",
993 certificate: ":new_certificate",
994 }
995
996 android_app_certificate {
997 name: "new_certificate",
998 certificate: "cert/new_cert",
999 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001000
1001 override_android_app {
1002 name: "baz",
1003 base: "foo",
1004 package_name: "org.dandroid.bp",
1005 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001006 `)
1007
1008 expectedVariants := []struct {
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001009 moduleName string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001010 variantName string
1011 apkName string
1012 apkPath string
1013 signFlag string
1014 overrides []string
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001015 aaptFlag string
Jaewoong Jung525443a2019-02-28 15:35:54 -08001016 }{
1017 {
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001018 moduleName: "foo",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001019 variantName: "android_common",
1020 apkPath: "/target/product/test_device/system/app/foo/foo.apk",
Dan Willemsen412160e2019-04-09 21:36:26 -07001021 signFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001022 overrides: []string{"qux"},
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001023 aaptFlag: "",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001024 },
1025 {
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001026 moduleName: "bar",
1027 variantName: "android_common_bar",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001028 apkPath: "/target/product/test_device/system/app/bar/bar.apk",
1029 signFlag: "cert/new_cert.x509.pem cert/new_cert.pk8",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001030 overrides: []string{"qux", "foo"},
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001031 aaptFlag: "",
1032 },
1033 {
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001034 moduleName: "baz",
1035 variantName: "android_common_baz",
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001036 apkPath: "/target/product/test_device/system/app/baz/baz.apk",
Dan Willemsen412160e2019-04-09 21:36:26 -07001037 signFlag: "build/make/target/product/security/expiredkey.x509.pem build/make/target/product/security/expiredkey.pk8",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001038 overrides: []string{"qux", "foo"},
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001039 aaptFlag: "--rename-manifest-package org.dandroid.bp",
Jaewoong Jung525443a2019-02-28 15:35:54 -08001040 },
1041 }
1042 for _, expected := range expectedVariants {
1043 variant := ctx.ModuleForTests("foo", expected.variantName)
1044
1045 // Check the final apk name
1046 outputs := variant.AllOutputs()
1047 expectedApkPath := buildDir + expected.apkPath
1048 found := false
1049 for _, o := range outputs {
1050 if o == expectedApkPath {
1051 found = true
1052 break
1053 }
1054 }
1055 if !found {
1056 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1057 }
1058
1059 // Check the certificate paths
Jaewoong Jung5a498812019-11-07 14:14:38 -08001060 signapk := variant.Output(expected.moduleName + ".apk")
Jaewoong Jung525443a2019-02-28 15:35:54 -08001061 signFlag := signapk.Args["certificates"]
1062 if expected.signFlag != signFlag {
1063 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected.signFlag, signFlag)
1064 }
1065
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001066 // Check if the overrides field values are correctly aggregated.
Jaewoong Jung525443a2019-02-28 15:35:54 -08001067 mod := variant.Module().(*AndroidApp)
1068 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1069 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1070 expected.overrides, mod.appProperties.Overrides)
1071 }
Jaewoong Jung6f373f62019-03-13 10:13:24 -07001072
1073 // Check the package renaming flag, if exists.
1074 res := variant.Output("package-res.apk")
1075 aapt2Flags := res.Args["flags"]
1076 if !strings.Contains(aapt2Flags, expected.aaptFlag) {
1077 t.Errorf("package renaming flag, %q is missing in aapt2 link flags, %q", expected.aaptFlag, aapt2Flags)
1078 }
Jaewoong Jung525443a2019-02-28 15:35:54 -08001079 }
1080}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001081
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001082func TestOverrideAndroidAppDependency(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001083 ctx, _ := testJava(t, `
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001084 android_app {
1085 name: "foo",
1086 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001087 sdk_version: "current",
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001088 }
1089
1090 override_android_app {
1091 name: "bar",
1092 base: "foo",
1093 package_name: "org.dandroid.bp",
1094 }
1095
1096 android_test {
1097 name: "baz",
1098 srcs: ["b.java"],
1099 instrumentation_for: "foo",
1100 }
1101
1102 android_test {
1103 name: "qux",
1104 srcs: ["b.java"],
1105 instrumentation_for: "bar",
1106 }
1107 `)
1108
1109 // Verify baz, which depends on the overridden module foo, has the correct classpath javac arg.
1110 javac := ctx.ModuleForTests("baz", "android_common").Rule("javac")
1111 fooTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common", "turbine-combined", "foo.jar")
1112 if !strings.Contains(javac.Args["classpath"], fooTurbine) {
1113 t.Errorf("baz classpath %v does not contain %q", javac.Args["classpath"], fooTurbine)
1114 }
1115
1116 // Verify qux, which depends on the overriding module bar, has the correct classpath javac arg.
1117 javac = ctx.ModuleForTests("qux", "android_common").Rule("javac")
1118 barTurbine := filepath.Join(buildDir, ".intermediates", "foo", "android_common_bar", "turbine-combined", "foo.jar")
1119 if !strings.Contains(javac.Args["classpath"], barTurbine) {
1120 t.Errorf("qux classpath %v does not contain %q", javac.Args["classpath"], barTurbine)
1121 }
1122}
1123
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001124func TestOverrideAndroidTest(t *testing.T) {
1125 ctx, _ := testJava(t, `
1126 android_app {
1127 name: "foo",
1128 srcs: ["a.java"],
1129 package_name: "com.android.foo",
1130 sdk_version: "current",
1131 }
1132
1133 override_android_app {
1134 name: "bar",
1135 base: "foo",
1136 package_name: "com.android.bar",
1137 }
1138
1139 android_test {
1140 name: "foo_test",
1141 srcs: ["b.java"],
1142 instrumentation_for: "foo",
1143 }
1144
1145 override_android_test {
1146 name: "bar_test",
1147 base: "foo_test",
1148 package_name: "com.android.bar.test",
1149 instrumentation_for: "bar",
1150 instrumentation_target_package: "com.android.bar",
1151 }
1152 `)
1153
1154 expectedVariants := []struct {
1155 moduleName string
1156 variantName string
1157 apkPath string
1158 overrides []string
1159 targetVariant string
1160 packageFlag string
1161 targetPackageFlag string
1162 }{
1163 {
1164 variantName: "android_common",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001165 apkPath: "/target/product/test_device/testcases/foo_test/arm64/foo_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001166 overrides: nil,
1167 targetVariant: "android_common",
1168 packageFlag: "",
1169 targetPackageFlag: "",
1170 },
1171 {
1172 variantName: "android_common_bar_test",
Jaewoong Jung326a9412019-11-21 10:41:00 -08001173 apkPath: "/target/product/test_device/testcases/bar_test/arm64/bar_test.apk",
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001174 overrides: []string{"foo_test"},
1175 targetVariant: "android_common_bar",
1176 packageFlag: "com.android.bar.test",
1177 targetPackageFlag: "com.android.bar",
1178 },
1179 }
1180 for _, expected := range expectedVariants {
1181 variant := ctx.ModuleForTests("foo_test", expected.variantName)
1182
1183 // Check the final apk name
1184 outputs := variant.AllOutputs()
1185 expectedApkPath := buildDir + expected.apkPath
1186 found := false
1187 for _, o := range outputs {
1188 if o == expectedApkPath {
1189 found = true
1190 break
1191 }
1192 }
1193 if !found {
1194 t.Errorf("Can't find %q in output files.\nAll outputs:%v", expectedApkPath, outputs)
1195 }
1196
1197 // Check if the overrides field values are correctly aggregated.
1198 mod := variant.Module().(*AndroidTest)
1199 if !reflect.DeepEqual(expected.overrides, mod.appProperties.Overrides) {
1200 t.Errorf("Incorrect overrides property value, expected: %q, got: %q",
1201 expected.overrides, mod.appProperties.Overrides)
1202 }
1203
1204 // Check if javac classpath has the correct jar file path. This checks instrumentation_for overrides.
1205 javac := variant.Rule("javac")
1206 turbine := filepath.Join(buildDir, ".intermediates", "foo", expected.targetVariant, "turbine-combined", "foo.jar")
1207 if !strings.Contains(javac.Args["classpath"], turbine) {
1208 t.Errorf("classpath %q does not contain %q", javac.Args["classpath"], turbine)
1209 }
1210
1211 // Check aapt2 flags.
1212 res := variant.Output("package-res.apk")
1213 aapt2Flags := res.Args["flags"]
1214 checkAapt2LinkFlag(t, aapt2Flags, "rename-manifest-package", expected.packageFlag)
1215 checkAapt2LinkFlag(t, aapt2Flags, "rename-instrumentation-target-package", expected.targetPackageFlag)
1216 }
1217}
1218
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001219func TestAndroidAppImport(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001220 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001221 android_app_import {
1222 name: "foo",
1223 apk: "prebuilts/apk/app.apk",
1224 certificate: "platform",
1225 dex_preopt: {
1226 enabled: true,
1227 },
1228 }
1229 `)
1230
1231 variant := ctx.ModuleForTests("foo", "android_common")
1232
1233 // Check dexpreopt outputs.
1234 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1235 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1236 t.Errorf("can't find dexpreopt outputs")
1237 }
1238
1239 // Check cert signing flag.
1240 signedApk := variant.Output("signed/foo.apk")
1241 signingFlag := signedApk.Args["certificates"]
1242 expected := "build/make/target/product/security/platform.x509.pem build/make/target/product/security/platform.pk8"
1243 if expected != signingFlag {
1244 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1245 }
1246}
1247
1248func TestAndroidAppImport_NoDexPreopt(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001249 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001250 android_app_import {
1251 name: "foo",
1252 apk: "prebuilts/apk/app.apk",
1253 certificate: "platform",
1254 dex_preopt: {
1255 enabled: false,
1256 },
1257 }
1258 `)
1259
1260 variant := ctx.ModuleForTests("foo", "android_common")
1261
1262 // Check dexpreopt outputs. They shouldn't exist.
1263 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule != nil ||
1264 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule != nil {
1265 t.Errorf("dexpreopt shouldn't have run.")
1266 }
1267}
1268
1269func TestAndroidAppImport_Presigned(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001270 ctx, _ := testJava(t, `
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001271 android_app_import {
1272 name: "foo",
1273 apk: "prebuilts/apk/app.apk",
1274 presigned: true,
1275 dex_preopt: {
1276 enabled: true,
1277 },
1278 }
1279 `)
1280
1281 variant := ctx.ModuleForTests("foo", "android_common")
1282
1283 // Check dexpreopt outputs.
1284 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1285 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1286 t.Errorf("can't find dexpreopt outputs")
1287 }
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +01001288 // Make sure signing was skipped and aligning was done.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001289 if variant.MaybeOutput("signed/foo.apk").Rule != nil {
1290 t.Errorf("signing rule shouldn't be included.")
1291 }
1292 if variant.MaybeOutput("zip-aligned/foo.apk").Rule == nil {
1293 t.Errorf("can't find aligning rule")
1294 }
1295}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001296
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001297func TestAndroidAppImport_DefaultDevCert(t *testing.T) {
1298 ctx, _ := testJava(t, `
1299 android_app_import {
1300 name: "foo",
1301 apk: "prebuilts/apk/app.apk",
1302 default_dev_cert: true,
1303 dex_preopt: {
1304 enabled: true,
1305 },
1306 }
1307 `)
1308
1309 variant := ctx.ModuleForTests("foo", "android_common")
1310
1311 // Check dexpreopt outputs.
1312 if variant.MaybeOutput("dexpreopt/oat/arm64/package.vdex").Rule == nil ||
1313 variant.MaybeOutput("dexpreopt/oat/arm64/package.odex").Rule == nil {
1314 t.Errorf("can't find dexpreopt outputs")
1315 }
1316
1317 // Check cert signing flag.
1318 signedApk := variant.Output("signed/foo.apk")
1319 signingFlag := signedApk.Args["certificates"]
1320 expected := "build/make/target/product/security/testkey.x509.pem build/make/target/product/security/testkey.pk8"
1321 if expected != signingFlag {
1322 t.Errorf("Incorrect signing flags, expected: %q, got: %q", expected, signingFlag)
1323 }
1324}
1325
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001326func TestAndroidAppImport_DpiVariants(t *testing.T) {
1327 bp := `
1328 android_app_import {
1329 name: "foo",
1330 apk: "prebuilts/apk/app.apk",
1331 dpi_variants: {
1332 xhdpi: {
1333 apk: "prebuilts/apk/app_xhdpi.apk",
1334 },
1335 xxhdpi: {
1336 apk: "prebuilts/apk/app_xxhdpi.apk",
1337 },
1338 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001339 presigned: true,
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001340 dex_preopt: {
1341 enabled: true,
1342 },
1343 }
1344 `
1345 testCases := []struct {
1346 name string
1347 aaptPreferredConfig *string
1348 aaptPrebuiltDPI []string
1349 expected string
1350 }{
1351 {
1352 name: "no preferred",
1353 aaptPreferredConfig: nil,
1354 aaptPrebuiltDPI: []string{},
1355 expected: "prebuilts/apk/app.apk",
1356 },
1357 {
1358 name: "AAPTPreferredConfig matches",
1359 aaptPreferredConfig: proptools.StringPtr("xhdpi"),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001360 aaptPrebuiltDPI: []string{"xxhdpi", "ldpi"},
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001361 expected: "prebuilts/apk/app_xhdpi.apk",
1362 },
1363 {
1364 name: "AAPTPrebuiltDPI matches",
1365 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1366 aaptPrebuiltDPI: []string{"xxhdpi", "xhdpi"},
1367 expected: "prebuilts/apk/app_xxhdpi.apk",
1368 },
1369 {
1370 name: "non-first AAPTPrebuiltDPI matches",
1371 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1372 aaptPrebuiltDPI: []string{"ldpi", "xhdpi"},
1373 expected: "prebuilts/apk/app_xhdpi.apk",
1374 },
1375 {
1376 name: "no matches",
1377 aaptPreferredConfig: proptools.StringPtr("mdpi"),
1378 aaptPrebuiltDPI: []string{"ldpi", "xxxhdpi"},
1379 expected: "prebuilts/apk/app.apk",
1380 },
1381 }
1382
1383 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
1384 for _, test := range testCases {
1385 config := testConfig(nil)
1386 config.TestProductVariables.AAPTPreferredConfig = test.aaptPreferredConfig
1387 config.TestProductVariables.AAPTPrebuiltDPI = test.aaptPrebuiltDPI
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001388 ctx := testAppContext(bp, nil)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001389
1390 run(t, ctx, config)
1391
1392 variant := ctx.ModuleForTests("foo", "android_common")
1393 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
1394 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
1395 if len(matches) != 2 {
1396 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
1397 }
1398 if test.expected != matches[1] {
1399 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
1400 }
1401 }
1402}
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001403
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07001404func TestAndroidAppImport_Filename(t *testing.T) {
1405 ctx, config := testJava(t, `
1406 android_app_import {
1407 name: "foo",
1408 apk: "prebuilts/apk/app.apk",
1409 presigned: true,
1410 }
1411
1412 android_app_import {
1413 name: "bar",
1414 apk: "prebuilts/apk/app.apk",
1415 presigned: true,
1416 filename: "bar_sample.apk"
1417 }
1418 `)
1419
1420 testCases := []struct {
1421 name string
1422 expected string
1423 }{
1424 {
1425 name: "foo",
1426 expected: "foo.apk",
1427 },
1428 {
1429 name: "bar",
1430 expected: "bar_sample.apk",
1431 },
1432 }
1433
1434 for _, test := range testCases {
1435 variant := ctx.ModuleForTests(test.name, "android_common")
1436 if variant.MaybeOutput(test.expected).Rule == nil {
1437 t.Errorf("can't find output named %q - all outputs: %v", test.expected, variant.AllOutputs())
1438 }
1439
1440 a := variant.Module().(*AndroidAppImport)
1441 expectedValues := []string{test.expected}
1442 actualValues := android.AndroidMkEntriesForTest(
1443 t, config, "", a).EntryMap["LOCAL_INSTALLED_MODULE_STEM"]
1444 if !reflect.DeepEqual(actualValues, expectedValues) {
1445 t.Errorf("Incorrect LOCAL_INSTALLED_MODULE_STEM value '%s', expected '%s'",
1446 actualValues, expectedValues)
1447 }
1448 }
1449}
1450
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001451func TestAndroidAppImport_ArchVariants(t *testing.T) {
1452 // The test config's target arch is ARM64.
1453 testCases := []struct {
1454 name string
1455 bp string
1456 expected string
1457 }{
1458 {
1459 name: "matching arch",
1460 bp: `
1461 android_app_import {
1462 name: "foo",
1463 apk: "prebuilts/apk/app.apk",
1464 arch: {
1465 arm64: {
1466 apk: "prebuilts/apk/app_arm64.apk",
1467 },
1468 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001469 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001470 dex_preopt: {
1471 enabled: true,
1472 },
1473 }
1474 `,
1475 expected: "prebuilts/apk/app_arm64.apk",
1476 },
1477 {
1478 name: "no matching arch",
1479 bp: `
1480 android_app_import {
1481 name: "foo",
1482 apk: "prebuilts/apk/app.apk",
1483 arch: {
1484 arm: {
1485 apk: "prebuilts/apk/app_arm.apk",
1486 },
1487 },
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001488 presigned: true,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001489 dex_preopt: {
1490 enabled: true,
1491 },
1492 }
1493 `,
1494 expected: "prebuilts/apk/app.apk",
1495 },
1496 }
1497
1498 jniRuleRe := regexp.MustCompile("^if \\(zipinfo (\\S+)")
1499 for _, test := range testCases {
1500 ctx, _ := testJava(t, test.bp)
1501
1502 variant := ctx.ModuleForTests("foo", "android_common")
1503 jniRuleCommand := variant.Output("jnis-uncompressed/foo.apk").RuleParams.Command
1504 matches := jniRuleRe.FindStringSubmatch(jniRuleCommand)
1505 if len(matches) != 2 {
1506 t.Errorf("failed to extract the src apk path from %q", jniRuleCommand)
1507 }
1508 if test.expected != matches[1] {
1509 t.Errorf("wrong src apk, expected: %q got: %q", test.expected, matches[1])
1510 }
1511 }
1512}
1513
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001514func TestAndroidTestImport(t *testing.T) {
1515 ctx, config := testJava(t, `
1516 android_test_import {
1517 name: "foo",
1518 apk: "prebuilts/apk/app.apk",
1519 presigned: true,
1520 data: [
1521 "testdata/data",
1522 ],
1523 }
1524 `)
1525
1526 test := ctx.ModuleForTests("foo", "android_common").Module().(*AndroidTestImport)
1527
1528 // Check android mks.
1529 entries := android.AndroidMkEntriesForTest(t, config, "", test)
1530 expected := []string{"tests"}
1531 actual := entries.EntryMap["LOCAL_MODULE_TAGS"]
1532 if !reflect.DeepEqual(expected, actual) {
1533 t.Errorf("Unexpected module tags - expected: %q, actual: %q", expected, actual)
1534 }
1535 expected = []string{"testdata/data:testdata/data"}
1536 actual = entries.EntryMap["LOCAL_COMPATIBILITY_SUPPORT_FILES"]
1537 if !reflect.DeepEqual(expected, actual) {
1538 t.Errorf("Unexpected test data - expected: %q, actual: %q", expected, actual)
1539 }
1540}
1541
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001542func TestStl(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001543 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001544 cc_library {
1545 name: "libjni",
1546 }
1547
1548 android_test {
1549 name: "stl",
1550 jni_libs: ["libjni"],
1551 compile_multilib: "both",
1552 sdk_version: "current",
1553 stl: "c++_shared",
1554 }
1555
1556 android_test {
1557 name: "system",
1558 jni_libs: ["libjni"],
1559 compile_multilib: "both",
1560 sdk_version: "current",
1561 }
Jaewoong Jung710756a2019-06-04 11:53:47 -07001562
1563 ndk_prebuilt_shared_stl {
1564 name: "ndk_libc++_shared",
1565 }
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001566 `)
1567
1568 testCases := []struct {
1569 name string
1570 jnis []string
1571 }{
1572 {"stl",
1573 []string{
1574 "libjni.so",
Jaewoong Jung710756a2019-06-04 11:53:47 -07001575 "libc++_shared.so",
Jaewoong Jungbc625cd2019-05-06 15:48:44 -07001576 },
1577 },
1578 {"system",
1579 []string{
1580 "libjni.so",
1581 },
1582 },
1583 }
1584
1585 for _, test := range testCases {
1586 t.Run(test.name, func(t *testing.T) {
1587 app := ctx.ModuleForTests(test.name, "android_common")
1588 jniLibZip := app.Output("jnilibs.zip")
1589 var jnis []string
1590 args := strings.Fields(jniLibZip.Args["jarArgs"])
1591 for i := 0; i < len(args); i++ {
1592 if args[i] == "-f" {
1593 jnis = append(jnis, args[i+1])
1594 i += 1
1595 }
1596 }
1597 jnisJoined := strings.Join(jnis, " ")
1598 for _, jni := range test.jnis {
1599 if !strings.Contains(jnisJoined, jni) {
1600 t.Errorf("missing jni %q in %q", jni, jnis)
1601 }
1602 }
1603 })
1604 }
1605}
Colin Cross50ddcc42019-05-16 12:28:22 -07001606
1607func TestUsesLibraries(t *testing.T) {
1608 bp := `
1609 java_sdk_library {
1610 name: "foo",
1611 srcs: ["a.java"],
1612 api_packages: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001613 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07001614 }
1615
1616 java_sdk_library {
1617 name: "bar",
1618 srcs: ["a.java"],
1619 api_packages: ["bar"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001620 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07001621 }
1622
1623 android_app {
1624 name: "app",
1625 srcs: ["a.java"],
1626 uses_libs: ["foo"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001627 sdk_version: "current",
Colin Cross50ddcc42019-05-16 12:28:22 -07001628 optional_uses_libs: [
1629 "bar",
1630 "baz",
1631 ],
1632 }
1633
1634 android_app_import {
1635 name: "prebuilt",
1636 apk: "prebuilts/apk/app.apk",
1637 certificate: "platform",
1638 uses_libs: ["foo"],
1639 optional_uses_libs: [
1640 "bar",
1641 "baz",
1642 ],
1643 }
1644 `
1645
1646 config := testConfig(nil)
1647 config.TestProductVariables.MissingUsesLibraries = []string{"baz"}
1648
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001649 ctx := testAppContext(bp, nil)
Colin Cross50ddcc42019-05-16 12:28:22 -07001650
1651 run(t, ctx, config)
1652
1653 app := ctx.ModuleForTests("app", "android_common")
1654 prebuilt := ctx.ModuleForTests("prebuilt", "android_common")
1655
1656 // Test that all libraries are verified
1657 cmd := app.Rule("verify_uses_libraries").RuleParams.Command
1658 if w := "--uses-library foo"; !strings.Contains(cmd, w) {
1659 t.Errorf("wanted %q in %q", w, cmd)
1660 }
1661
1662 if w := "--optional-uses-library bar --optional-uses-library baz"; !strings.Contains(cmd, w) {
1663 t.Errorf("wanted %q in %q", w, cmd)
1664 }
1665
1666 cmd = prebuilt.Rule("verify_uses_libraries").RuleParams.Command
1667
1668 if w := `uses_library_names="foo"`; !strings.Contains(cmd, w) {
1669 t.Errorf("wanted %q in %q", w, cmd)
1670 }
1671
1672 if w := `optional_uses_library_names="bar baz"`; !strings.Contains(cmd, w) {
1673 t.Errorf("wanted %q in %q", w, cmd)
1674 }
1675
1676 // Test that only present libraries are preopted
1677 cmd = app.Rule("dexpreopt").RuleParams.Command
1678
1679 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
1680 t.Errorf("wanted %q in %q", w, cmd)
1681 }
1682
1683 cmd = prebuilt.Rule("dexpreopt").RuleParams.Command
1684
1685 if w := `dex_preopt_target_libraries="/system/framework/foo.jar /system/framework/bar.jar"`; !strings.Contains(cmd, w) {
1686 t.Errorf("wanted %q in %q", w, cmd)
1687 }
1688}
Jaewoong Jungc27ab662019-05-30 15:51:14 -07001689
1690func TestCodelessApp(t *testing.T) {
1691 testCases := []struct {
1692 name string
1693 bp string
1694 noCode bool
1695 }{
1696 {
1697 name: "normal",
1698 bp: `
1699 android_app {
1700 name: "foo",
1701 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001702 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07001703 }
1704 `,
1705 noCode: false,
1706 },
1707 {
1708 name: "app without sources",
1709 bp: `
1710 android_app {
1711 name: "foo",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001712 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07001713 }
1714 `,
1715 noCode: true,
1716 },
1717 {
1718 name: "app with libraries",
1719 bp: `
1720 android_app {
1721 name: "foo",
1722 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001723 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07001724 }
1725
1726 java_library {
1727 name: "lib",
1728 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001729 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07001730 }
1731 `,
1732 noCode: false,
1733 },
1734 {
1735 name: "app with sourceless libraries",
1736 bp: `
1737 android_app {
1738 name: "foo",
1739 static_libs: ["lib"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001740 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07001741 }
1742
1743 java_library {
1744 name: "lib",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001745 sdk_version: "current",
Jaewoong Jungc27ab662019-05-30 15:51:14 -07001746 }
1747 `,
1748 // TODO(jungjw): this should probably be true
1749 noCode: false,
1750 },
1751 }
1752
1753 for _, test := range testCases {
1754 t.Run(test.name, func(t *testing.T) {
1755 ctx := testApp(t, test.bp)
1756
1757 foo := ctx.ModuleForTests("foo", "android_common")
1758 manifestFixerArgs := foo.Output("manifest_fixer/AndroidManifest.xml").Args["args"]
1759 if strings.Contains(manifestFixerArgs, "--has-no-code") != test.noCode {
1760 t.Errorf("unexpected manifest_fixer args: %q", manifestFixerArgs)
1761 }
1762 })
1763 }
1764}
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001765
1766func TestEmbedNotice(t *testing.T) {
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001767 ctx, _ := testJava(t, cc.GatherRequiredDepsForTest(android.Android)+`
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001768 android_app {
1769 name: "foo",
1770 srcs: ["a.java"],
1771 static_libs: ["javalib"],
1772 jni_libs: ["libjni"],
1773 notice: "APP_NOTICE",
1774 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001775 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001776 }
1777
1778 // No embed_notice flag
1779 android_app {
1780 name: "bar",
1781 srcs: ["a.java"],
1782 jni_libs: ["libjni"],
1783 notice: "APP_NOTICE",
Jeongik Cha538c0d02019-07-11 15:54:27 +09001784 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001785 }
1786
1787 // No NOTICE files
1788 android_app {
1789 name: "baz",
1790 srcs: ["a.java"],
1791 embed_notices: true,
Jeongik Cha538c0d02019-07-11 15:54:27 +09001792 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001793 }
1794
1795 cc_library {
1796 name: "libjni",
1797 system_shared_libs: [],
1798 stl: "none",
1799 notice: "LIB_NOTICE",
1800 }
1801
1802 java_library {
1803 name: "javalib",
1804 srcs: [
1805 ":gen",
1806 ],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001807 sdk_version: "current",
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001808 }
1809
1810 genrule {
1811 name: "gen",
1812 tools: ["gentool"],
1813 out: ["gen.java"],
1814 notice: "GENRULE_NOTICE",
1815 }
1816
1817 java_binary_host {
1818 name: "gentool",
1819 srcs: ["b.java"],
1820 notice: "TOOL_NOTICE",
1821 }
1822 `)
1823
1824 // foo has NOTICE files to process, and embed_notices is true.
1825 foo := ctx.ModuleForTests("foo", "android_common")
1826 // verify merge notices rule.
1827 mergeNotices := foo.Rule("mergeNoticesRule")
1828 noticeInputs := mergeNotices.Inputs.Strings()
1829 // TOOL_NOTICE should be excluded as it's a host module.
1830 if len(mergeNotices.Inputs) != 3 {
1831 t.Errorf("number of input notice files: expected = 3, actual = %q", noticeInputs)
1832 }
1833 if !inList("APP_NOTICE", noticeInputs) {
1834 t.Errorf("APP_NOTICE is missing from notice files, %q", noticeInputs)
1835 }
1836 if !inList("LIB_NOTICE", noticeInputs) {
1837 t.Errorf("LIB_NOTICE is missing from notice files, %q", noticeInputs)
1838 }
1839 if !inList("GENRULE_NOTICE", noticeInputs) {
1840 t.Errorf("GENRULE_NOTICE is missing from notice files, %q", noticeInputs)
1841 }
1842 // aapt2 flags should include -A <NOTICE dir> so that its contents are put in the APK's /assets.
1843 res := foo.Output("package-res.apk")
1844 aapt2Flags := res.Args["flags"]
1845 e := "-A " + buildDir + "/.intermediates/foo/android_common/NOTICE"
1846 if !strings.Contains(aapt2Flags, e) {
1847 t.Errorf("asset dir flag for NOTICE, %q is missing in aapt2 link flags, %q", e, aapt2Flags)
1848 }
1849
1850 // bar has NOTICE files to process, but embed_notices is not set.
1851 bar := ctx.ModuleForTests("bar", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07001852 res = bar.Output("package-res.apk")
1853 aapt2Flags = res.Args["flags"]
1854 e = "-A " + buildDir + "/.intermediates/bar/android_common/NOTICE"
1855 if strings.Contains(aapt2Flags, e) {
1856 t.Errorf("bar shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001857 }
1858
1859 // baz's embed_notice is true, but it doesn't have any NOTICE files.
1860 baz := ctx.ModuleForTests("baz", "android_common")
Jaewoong Jung98772792019-07-01 17:15:13 -07001861 res = baz.Output("package-res.apk")
1862 aapt2Flags = res.Args["flags"]
1863 e = "-A " + buildDir + "/.intermediates/baz/android_common/NOTICE"
1864 if strings.Contains(aapt2Flags, e) {
1865 t.Errorf("baz shouldn't have the asset dir flag for NOTICE: %q", e)
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001866 }
1867}
Colin Cross53a87f52019-06-25 13:35:30 -07001868
1869func TestUncompressDex(t *testing.T) {
1870 testCases := []struct {
1871 name string
1872 bp string
1873
1874 uncompressedPlatform bool
1875 uncompressedUnbundled bool
1876 }{
1877 {
1878 name: "normal",
1879 bp: `
1880 android_app {
1881 name: "foo",
1882 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001883 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07001884 }
1885 `,
1886 uncompressedPlatform: true,
1887 uncompressedUnbundled: false,
1888 },
1889 {
1890 name: "use_embedded_dex",
1891 bp: `
1892 android_app {
1893 name: "foo",
1894 use_embedded_dex: true,
1895 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001896 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07001897 }
1898 `,
1899 uncompressedPlatform: true,
1900 uncompressedUnbundled: true,
1901 },
1902 {
1903 name: "privileged",
1904 bp: `
1905 android_app {
1906 name: "foo",
1907 privileged: true,
1908 srcs: ["a.java"],
Jeongik Cha538c0d02019-07-11 15:54:27 +09001909 sdk_version: "current",
Colin Cross53a87f52019-06-25 13:35:30 -07001910 }
1911 `,
1912 uncompressedPlatform: true,
1913 uncompressedUnbundled: true,
1914 },
1915 }
1916
1917 test := func(t *testing.T, bp string, want bool, unbundled bool) {
1918 t.Helper()
1919
1920 config := testConfig(nil)
1921 if unbundled {
1922 config.TestProductVariables.Unbundled_build = proptools.BoolPtr(true)
1923 }
1924
Jaewoong Jungf9a04432019-07-17 11:15:09 -07001925 ctx := testAppContext(bp, nil)
Colin Cross53a87f52019-06-25 13:35:30 -07001926
1927 run(t, ctx, config)
1928
1929 foo := ctx.ModuleForTests("foo", "android_common")
1930 dex := foo.Rule("r8")
1931 uncompressedInDexJar := strings.Contains(dex.Args["zipFlags"], "-L 0")
1932 aligned := foo.MaybeRule("zipalign").Rule != nil
1933
1934 if uncompressedInDexJar != want {
1935 t.Errorf("want uncompressed in dex %v, got %v", want, uncompressedInDexJar)
1936 }
1937
1938 if aligned != want {
1939 t.Errorf("want aligned %v, got %v", want, aligned)
1940 }
1941 }
1942
1943 for _, tt := range testCases {
1944 t.Run(tt.name, func(t *testing.T) {
1945 t.Run("platform", func(t *testing.T) {
1946 test(t, tt.bp, tt.uncompressedPlatform, false)
1947 })
1948 t.Run("unbundled", func(t *testing.T) {
1949 test(t, tt.bp, tt.uncompressedUnbundled, true)
1950 })
1951 })
1952 }
1953}
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001954
1955func checkAapt2LinkFlag(t *testing.T, aapt2Flags, flagName, expectedValue string) {
1956 if expectedValue != "" {
1957 expectedFlag := "--" + flagName + " " + expectedValue
1958 if !strings.Contains(aapt2Flags, expectedFlag) {
1959 t.Errorf("%q is missing in aapt2 link flags, %q", expectedFlag, aapt2Flags)
1960 }
1961 } else {
1962 unexpectedFlag := "--" + flagName
1963 if strings.Contains(aapt2Flags, unexpectedFlag) {
1964 t.Errorf("unexpected flag, %q is found in aapt2 link flags, %q", unexpectedFlag, aapt2Flags)
1965 }
1966 }
1967}