blob: 47c64a9ad43520d73833b1faf85fec03e074ced2 [file] [log] [blame]
Ivan Lozanoffee3342019-08-27 12:03:00 -07001// Copyright 2019 The Android Open Source Project
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 rust
16
17import (
Ivan Lozanoffee3342019-08-27 12:03:00 -070018 "os"
Ivan Lozanoc0083612019-09-03 13:49:39 -070019 "runtime"
Ivan Lozanob9040d62019-09-24 13:23:50 -070020 "strings"
Ivan Lozanoffee3342019-08-27 12:03:00 -070021 "testing"
22
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -040023 "github.com/google/blueprint/proptools"
24
Ivan Lozanoffee3342019-08-27 12:03:00 -070025 "android/soong/android"
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000026 "android/soong/genrule"
Ivan Lozanoffee3342019-08-27 12:03:00 -070027)
28
Ivan Lozanoffee3342019-08-27 12:03:00 -070029func TestMain(m *testing.M) {
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000030 os.Exit(m.Run())
31}
Ivan Lozanoffee3342019-08-27 12:03:00 -070032
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000033var prepareForRustTest = android.GroupFixturePreparers(
34 android.PrepareForTestWithArchMutator,
35 android.PrepareForTestWithDefaults,
36 android.PrepareForTestWithPrebuilts,
Ivan Lozanoffee3342019-08-27 12:03:00 -070037
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000038 genrule.PrepareForTestWithGenRuleBuildComponents,
39
40 PrepareForIntegrationTestWithRust,
41)
42
43var rustMockedFiles = android.MockFS{
44 "foo.rs": nil,
45 "foo.c": nil,
46 "src/bar.rs": nil,
47 "src/any.h": nil,
48 "proto.proto": nil,
49 "proto/buf.proto": nil,
50 "buf.proto": nil,
51 "foo.proto": nil,
52 "liby.so": nil,
53 "libz.so": nil,
54 "data.txt": nil,
Ivan Lozanoffee3342019-08-27 12:03:00 -070055}
56
Thiébaud Weksteen0a75e522020-10-07 14:30:03 +020057// testRust returns a TestContext in which a basic environment has been setup.
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000058// This environment contains a few mocked files. See rustMockedFiles for the list of these files.
Thiébaud Weksteen0a75e522020-10-07 14:30:03 +020059func testRust(t *testing.T, bp string) *android.TestContext {
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000060 skipTestIfOsNotSupported(t)
61 result := android.GroupFixturePreparers(
62 prepareForRustTest,
63 rustMockedFiles.AddToFixture(),
64 ).
65 RunTestWithBp(t, bp)
66 return result.TestContext
Thiébaud Weksteen0a75e522020-10-07 14:30:03 +020067}
Colin Cross98be1bb2019-12-13 20:41:13 -080068
Ivan Lozanof76cdf72021-02-12 09:55:06 -050069func testRustVndk(t *testing.T, bp string) *android.TestContext {
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000070 skipTestIfOsNotSupported(t)
71 result := android.GroupFixturePreparers(
72 prepareForRustTest,
73 rustMockedFiles.AddToFixture(),
74 android.FixtureModifyProductVariables(
75 func(variables android.FixtureProductVariables) {
76 variables.DeviceVndkVersion = StringPtr("current")
77 variables.ProductVndkVersion = StringPtr("current")
Jiyong Parkf58c46e2021-04-01 21:35:20 +090078 variables.Platform_vndk_version = StringPtr("29")
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000079 },
80 ),
81 ).RunTestWithBp(t, bp)
82 return result.TestContext
Ivan Lozanof76cdf72021-02-12 09:55:06 -050083}
84
Thiébaud Weksteen0a75e522020-10-07 14:30:03 +020085// testRustCov returns a TestContext in which a basic environment has been
86// setup. This environment explicitly enables coverage.
87func testRustCov(t *testing.T, bp string) *android.TestContext {
Paul Duffin2c4ca8d2021-03-07 19:18:38 +000088 skipTestIfOsNotSupported(t)
89 result := android.GroupFixturePreparers(
90 prepareForRustTest,
91 rustMockedFiles.AddToFixture(),
92 android.FixtureModifyProductVariables(
93 func(variables android.FixtureProductVariables) {
94 variables.ClangCoverage = proptools.BoolPtr(true)
95 variables.Native_coverage = proptools.BoolPtr(true)
96 variables.NativeCoveragePaths = []string{"*"}
97 },
98 ),
99 ).RunTestWithBp(t, bp)
100 return result.TestContext
Thiébaud Weksteen0a75e522020-10-07 14:30:03 +0200101}
102
103// testRustError ensures that at least one error was raised and its value
104// matches the pattern provided. The error can be either in the parsing of the
105// Blueprint or when generating the build actions.
106func testRustError(t *testing.T, pattern string, bp string) {
Paul Duffin2c4ca8d2021-03-07 19:18:38 +0000107 skipTestIfOsNotSupported(t)
108 android.GroupFixturePreparers(
109 prepareForRustTest,
110 rustMockedFiles.AddToFixture(),
111 ).
112 ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(pattern)).
113 RunTestWithBp(t, bp)
Thiébaud Weksteen0a75e522020-10-07 14:30:03 +0200114}
115
Ivan Lozanoc08897c2021-04-02 12:41:32 -0400116// testRustVndkError is similar to testRustError, but can be used to test VNDK-related errors.
117func testRustVndkError(t *testing.T, pattern string, bp string) {
118 skipTestIfOsNotSupported(t)
119 android.GroupFixturePreparers(
120 prepareForRustTest,
121 rustMockedFiles.AddToFixture(),
122 android.FixtureModifyProductVariables(
123 func(variables android.FixtureProductVariables) {
124 variables.DeviceVndkVersion = StringPtr("current")
125 variables.ProductVndkVersion = StringPtr("current")
126 variables.Platform_vndk_version = StringPtr("VER")
127 },
128 ),
129 ).
130 ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(pattern)).
131 RunTestWithBp(t, bp)
132}
133
Thiébaud Weksteen0a75e522020-10-07 14:30:03 +0200134// testRustCtx is used to build a particular test environment. Unless your
135// tests requires a specific setup, prefer the wrapping functions: testRust,
136// testRustCov or testRustError.
137type testRustCtx struct {
138 bp string
139 fs map[string][]byte
140 env map[string]string
141 config *android.Config
142}
143
Paul Duffin2c4ca8d2021-03-07 19:18:38 +0000144func skipTestIfOsNotSupported(t *testing.T) {
Thiébaud Weksteen0a75e522020-10-07 14:30:03 +0200145 // TODO (b/140435149)
146 if runtime.GOOS != "linux" {
147 t.Skip("Rust Soong tests can only be run on Linux hosts currently")
148 }
Ivan Lozanoffee3342019-08-27 12:03:00 -0700149}
150
Ivan Lozanoffee3342019-08-27 12:03:00 -0700151// Test that we can extract the link path from a lib path.
152func TestLinkPathFromFilePath(t *testing.T) {
153 barPath := android.PathForTesting("out/soong/.intermediates/external/libbar/libbar/linux_glibc_x86_64_shared/libbar.so")
154 libName := linkPathFromFilePath(barPath)
155 expectedResult := "out/soong/.intermediates/external/libbar/libbar/linux_glibc_x86_64_shared/"
156
157 if libName != expectedResult {
158 t.Errorf("libNameFromFilePath returned the wrong name; expected '%#v', got '%#v'", expectedResult, libName)
159 }
160}
161
Ivan Lozanoffee3342019-08-27 12:03:00 -0700162// Test to make sure dependencies are being picked up correctly.
163func TestDepsTracking(t *testing.T) {
164 ctx := testRust(t, `
Matthew Maurer2ae05132020-06-23 14:28:53 -0700165 rust_ffi_host_static {
Ivan Lozano52767be2019-10-18 14:49:46 -0700166 name: "libstatic",
167 srcs: ["foo.rs"],
Ivan Lozanoad8b18b2019-10-31 19:38:29 -0700168 crate_name: "static",
Ivan Lozano52767be2019-10-18 14:49:46 -0700169 }
Ivan Lozano63bb7682021-03-23 15:53:44 -0400170 rust_ffi_host_static {
171 name: "libwholestatic",
172 srcs: ["foo.rs"],
173 crate_name: "wholestatic",
174 }
Matthew Maurer2ae05132020-06-23 14:28:53 -0700175 rust_ffi_host_shared {
Ivan Lozano52767be2019-10-18 14:49:46 -0700176 name: "libshared",
177 srcs: ["foo.rs"],
Ivan Lozanoad8b18b2019-10-31 19:38:29 -0700178 crate_name: "shared",
Ivan Lozano52767be2019-10-18 14:49:46 -0700179 }
Ivan Lozanoffee3342019-08-27 12:03:00 -0700180 rust_library_host_dylib {
Ivan Lozano52767be2019-10-18 14:49:46 -0700181 name: "libdylib",
Ivan Lozanoffee3342019-08-27 12:03:00 -0700182 srcs: ["foo.rs"],
Ivan Lozanoad8b18b2019-10-31 19:38:29 -0700183 crate_name: "dylib",
Ivan Lozanoffee3342019-08-27 12:03:00 -0700184 }
185 rust_library_host_rlib {
Ivan Lozano52767be2019-10-18 14:49:46 -0700186 name: "librlib",
Ivan Lozano43845682020-07-09 21:03:28 -0400187 srcs: ["foo.rs"],
Ivan Lozanoad8b18b2019-10-31 19:38:29 -0700188 crate_name: "rlib",
Ivan Lozanofb6f36f2021-02-05 12:27:08 -0500189 static_libs: ["libstatic"],
Ivan Lozano63bb7682021-03-23 15:53:44 -0400190 whole_static_libs: ["libwholestatic"],
Ivan Lozanoffee3342019-08-27 12:03:00 -0700191 }
192 rust_proc_macro {
193 name: "libpm",
194 srcs: ["foo.rs"],
Ivan Lozanoad8b18b2019-10-31 19:38:29 -0700195 crate_name: "pm",
Ivan Lozanoffee3342019-08-27 12:03:00 -0700196 }
197 rust_binary_host {
Ivan Lozano43845682020-07-09 21:03:28 -0400198 name: "fizz-buzz",
Ivan Lozano52767be2019-10-18 14:49:46 -0700199 dylibs: ["libdylib"],
200 rlibs: ["librlib"],
Ivan Lozanoffee3342019-08-27 12:03:00 -0700201 proc_macros: ["libpm"],
Ivan Lozano52767be2019-10-18 14:49:46 -0700202 static_libs: ["libstatic"],
203 shared_libs: ["libshared"],
Ivan Lozano43845682020-07-09 21:03:28 -0400204 srcs: ["foo.rs"],
Ivan Lozanoffee3342019-08-27 12:03:00 -0700205 }
206 `)
Ivan Lozano43845682020-07-09 21:03:28 -0400207 module := ctx.ModuleForTests("fizz-buzz", "linux_glibc_x86_64").Module().(*Module)
Ivan Lozanofb6f36f2021-02-05 12:27:08 -0500208 rustc := ctx.ModuleForTests("librlib", "linux_glibc_x86_64_rlib_rlib-std").Rule("rustc")
Ivan Lozanoffee3342019-08-27 12:03:00 -0700209
210 // Since dependencies are added to AndroidMk* properties, we can check these to see if they've been picked up.
Ivan Lozano52767be2019-10-18 14:49:46 -0700211 if !android.InList("libdylib", module.Properties.AndroidMkDylibs) {
Ivan Lozanoffee3342019-08-27 12:03:00 -0700212 t.Errorf("Dylib dependency not detected (dependency missing from AndroidMkDylibs)")
213 }
214
Ivan Lozano2b081132020-09-08 12:46:52 -0400215 if !android.InList("librlib.rlib-std", module.Properties.AndroidMkRlibs) {
Ivan Lozanoffee3342019-08-27 12:03:00 -0700216 t.Errorf("Rlib dependency not detected (dependency missing from AndroidMkRlibs)")
217 }
218
219 if !android.InList("libpm", module.Properties.AndroidMkProcMacroLibs) {
220 t.Errorf("Proc_macro dependency not detected (dependency missing from AndroidMkProcMacroLibs)")
221 }
222
Ivan Lozano52767be2019-10-18 14:49:46 -0700223 if !android.InList("libshared", module.Properties.AndroidMkSharedLibs) {
224 t.Errorf("Shared library dependency not detected (dependency missing from AndroidMkSharedLibs)")
225 }
226
227 if !android.InList("libstatic", module.Properties.AndroidMkStaticLibs) {
228 t.Errorf("Static library dependency not detected (dependency missing from AndroidMkStaticLibs)")
229 }
Ivan Lozano3dfa12d2021-02-04 11:29:41 -0500230
Ivan Lozano63bb7682021-03-23 15:53:44 -0400231 if !strings.Contains(rustc.Args["rustcFlags"], "-lstatic=wholestatic") {
232 t.Errorf("-lstatic flag not being passed to rustc for static library %#v", rustc.Args["rustcFlags"])
Ivan Lozano3dfa12d2021-02-04 11:29:41 -0500233 }
234
Ivan Lozanoffee3342019-08-27 12:03:00 -0700235}
Ivan Lozanob9040d62019-09-24 13:23:50 -0700236
Ivan Lozano43845682020-07-09 21:03:28 -0400237func TestSourceProviderDeps(t *testing.T) {
238 ctx := testRust(t, `
239 rust_binary {
240 name: "fizz-buzz-dep",
241 srcs: [
242 "foo.rs",
243 ":my_generator",
244 ":libbindings",
245 ],
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400246 rlibs: ["libbindings"],
Ivan Lozano43845682020-07-09 21:03:28 -0400247 }
248 rust_proc_macro {
249 name: "libprocmacro",
250 srcs: [
251 "foo.rs",
252 ":my_generator",
253 ":libbindings",
254 ],
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400255 rlibs: ["libbindings"],
Ivan Lozano43845682020-07-09 21:03:28 -0400256 crate_name: "procmacro",
257 }
258 rust_library {
259 name: "libfoo",
260 srcs: [
261 "foo.rs",
262 ":my_generator",
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400263 ":libbindings",
264 ],
265 rlibs: ["libbindings"],
Ivan Lozano43845682020-07-09 21:03:28 -0400266 crate_name: "foo",
267 }
268 genrule {
269 name: "my_generator",
270 tools: ["any_rust_binary"],
271 cmd: "$(location) -o $(out) $(in)",
272 srcs: ["src/any.h"],
273 out: ["src/any.rs"],
274 }
Colin Crosse9fe2942020-11-10 18:12:15 -0800275 rust_binary_host {
276 name: "any_rust_binary",
277 srcs: [
278 "foo.rs",
279 ],
280 }
Ivan Lozano43845682020-07-09 21:03:28 -0400281 rust_bindgen {
282 name: "libbindings",
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400283 crate_name: "bindings",
284 source_stem: "bindings",
Ivan Lozano43845682020-07-09 21:03:28 -0400285 host_supported: true,
286 wrapper_src: "src/any.h",
287 }
288 `)
289
Ivan Lozano2b081132020-09-08 12:46:52 -0400290 libfoo := ctx.ModuleForTests("libfoo", "android_arm64_armv8-a_rlib_dylib-std").Rule("rustc")
Ivan Lozano43845682020-07-09 21:03:28 -0400291 if !android.SuffixInList(libfoo.Implicits.Strings(), "/out/bindings.rs") {
292 t.Errorf("rust_bindgen generated source not included as implicit input for libfoo; Implicits %#v", libfoo.Implicits.Strings())
293 }
294 if !android.SuffixInList(libfoo.Implicits.Strings(), "/out/any.rs") {
295 t.Errorf("genrule generated source not included as implicit input for libfoo; Implicits %#v", libfoo.Implicits.Strings())
296 }
297
298 fizzBuzz := ctx.ModuleForTests("fizz-buzz-dep", "android_arm64_armv8-a").Rule("rustc")
299 if !android.SuffixInList(fizzBuzz.Implicits.Strings(), "/out/bindings.rs") {
300 t.Errorf("rust_bindgen generated source not included as implicit input for fizz-buzz-dep; Implicits %#v", libfoo.Implicits.Strings())
301 }
302 if !android.SuffixInList(fizzBuzz.Implicits.Strings(), "/out/any.rs") {
303 t.Errorf("genrule generated source not included as implicit input for fizz-buzz-dep; Implicits %#v", libfoo.Implicits.Strings())
304 }
305
306 libprocmacro := ctx.ModuleForTests("libprocmacro", "linux_glibc_x86_64").Rule("rustc")
307 if !android.SuffixInList(libprocmacro.Implicits.Strings(), "/out/bindings.rs") {
308 t.Errorf("rust_bindgen generated source not included as implicit input for libprocmacro; Implicits %#v", libfoo.Implicits.Strings())
309 }
310 if !android.SuffixInList(libprocmacro.Implicits.Strings(), "/out/any.rs") {
311 t.Errorf("genrule generated source not included as implicit input for libprocmacro; Implicits %#v", libfoo.Implicits.Strings())
312 }
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400313
314 // Check that our bindings are picked up as crate dependencies as well
315 libfooMod := ctx.ModuleForTests("libfoo", "android_arm64_armv8-a_dylib").Module().(*Module)
Ivan Lozano2b081132020-09-08 12:46:52 -0400316 if !android.InList("libbindings.dylib-std", libfooMod.Properties.AndroidMkRlibs) {
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400317 t.Errorf("bindgen dependency not detected as a rlib dependency (dependency missing from AndroidMkRlibs)")
318 }
319 fizzBuzzMod := ctx.ModuleForTests("fizz-buzz-dep", "android_arm64_armv8-a").Module().(*Module)
Ivan Lozano2b081132020-09-08 12:46:52 -0400320 if !android.InList("libbindings.dylib-std", fizzBuzzMod.Properties.AndroidMkRlibs) {
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400321 t.Errorf("bindgen dependency not detected as a rlib dependency (dependency missing from AndroidMkRlibs)")
322 }
323 libprocmacroMod := ctx.ModuleForTests("libprocmacro", "linux_glibc_x86_64").Module().(*Module)
Ivan Lozano2b081132020-09-08 12:46:52 -0400324 if !android.InList("libbindings.rlib-std", libprocmacroMod.Properties.AndroidMkRlibs) {
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400325 t.Errorf("bindgen dependency not detected as a rlib dependency (dependency missing from AndroidMkRlibs)")
326 }
327
Ivan Lozano43845682020-07-09 21:03:28 -0400328}
329
Ivan Lozano07cbaf42020-07-22 16:09:13 -0400330func TestSourceProviderTargetMismatch(t *testing.T) {
331 // This might error while building the dependency tree or when calling depsToPaths() depending on the lunched
332 // target, which results in two different errors. So don't check the error, just confirm there is one.
333 testRustError(t, ".*", `
334 rust_proc_macro {
335 name: "libprocmacro",
336 srcs: [
337 "foo.rs",
338 ":libbindings",
339 ],
340 crate_name: "procmacro",
341 }
342 rust_bindgen {
343 name: "libbindings",
Ivan Lozano26ecd6c2020-07-31 13:40:31 -0400344 crate_name: "bindings",
345 source_stem: "bindings",
Ivan Lozano07cbaf42020-07-22 16:09:13 -0400346 wrapper_src: "src/any.h",
347 }
348 `)
349}
350
Ivan Lozanob9040d62019-09-24 13:23:50 -0700351// Test to make sure proc_macros use host variants when building device modules.
352func TestProcMacroDeviceDeps(t *testing.T) {
353 ctx := testRust(t, `
354 rust_library_host_rlib {
355 name: "libbar",
356 srcs: ["foo.rs"],
Ivan Lozanoad8b18b2019-10-31 19:38:29 -0700357 crate_name: "bar",
Ivan Lozanob9040d62019-09-24 13:23:50 -0700358 }
359 rust_proc_macro {
360 name: "libpm",
361 rlibs: ["libbar"],
362 srcs: ["foo.rs"],
Ivan Lozanoad8b18b2019-10-31 19:38:29 -0700363 crate_name: "pm",
Ivan Lozanob9040d62019-09-24 13:23:50 -0700364 }
365 rust_binary {
366 name: "fizz-buzz",
367 proc_macros: ["libpm"],
368 srcs: ["foo.rs"],
369 }
370 `)
371 rustc := ctx.ModuleForTests("libpm", "linux_glibc_x86_64").Rule("rustc")
372
373 if !strings.Contains(rustc.Args["libFlags"], "libbar/linux_glibc_x86_64") {
374 t.Errorf("Proc_macro is not using host variant of dependent modules.")
375 }
376}
Matthew Maurer99020b02019-10-31 10:44:40 -0700377
378// Test that no_stdlibs suppresses dependencies on rust standard libraries
379func TestNoStdlibs(t *testing.T) {
380 ctx := testRust(t, `
381 rust_binary {
382 name: "fizz-buzz",
383 srcs: ["foo.rs"],
Ivan Lozano9d1df102020-04-28 10:10:23 -0400384 no_stdlibs: true,
Matthew Maurer99020b02019-10-31 10:44:40 -0700385 }`)
Colin Cross7113d202019-11-20 16:39:12 -0800386 module := ctx.ModuleForTests("fizz-buzz", "android_arm64_armv8-a").Module().(*Module)
Matthew Maurer99020b02019-10-31 10:44:40 -0700387
388 if android.InList("libstd", module.Properties.AndroidMkDylibs) {
389 t.Errorf("no_stdlibs did not suppress dependency on libstd")
390 }
391}
Ivan Lozano9d1df102020-04-28 10:10:23 -0400392
393// Test that libraries provide both 32-bit and 64-bit variants.
394func TestMultilib(t *testing.T) {
395 ctx := testRust(t, `
396 rust_library_rlib {
397 name: "libfoo",
398 srcs: ["foo.rs"],
399 crate_name: "foo",
400 }`)
401
Ivan Lozano2b081132020-09-08 12:46:52 -0400402 _ = ctx.ModuleForTests("libfoo", "android_arm64_armv8-a_rlib_dylib-std")
403 _ = ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_rlib_dylib-std")
Ivan Lozano9d1df102020-04-28 10:10:23 -0400404}