| // Copyright 2019 The Android Open Source Project |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package rust |
| |
| import ( |
| "strings" |
| "testing" |
| |
| "android/soong/android" |
| ) |
| |
| // Test that rustlibs default linkage is always rlib for host binaries. |
| func TestBinaryHostLinkage(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary_host { |
| name: "fizz-buzz", |
| srcs: ["foo.rs"], |
| rustlibs: ["libfoo"], |
| } |
| rust_library { |
| name: "libfoo", |
| srcs: ["foo.rs"], |
| crate_name: "foo", |
| host_supported: true, |
| } |
| `) |
| fizzBuzz := ctx.ModuleForTests("fizz-buzz", "linux_glibc_x86_64").Module().(*Module) |
| if !android.InList("libfoo.rlib-std", fizzBuzz.Properties.AndroidMkRlibs) { |
| t.Errorf("rustlibs dependency libfoo should be an rlib dep for host binaries") |
| } |
| } |
| |
| // Test that rustlibs default linkage is correct for binaries. |
| func TestBinaryLinkage(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary { |
| name: "fizz-buzz", |
| srcs: ["foo.rs"], |
| rustlibs: ["libfoo"], |
| host_supported: true, |
| } |
| rust_binary { |
| name: "rlib_linked", |
| srcs: ["foo.rs"], |
| rustlibs: ["libfoo"], |
| host_supported: true, |
| prefer_rlib: true, |
| } |
| rust_library { |
| name: "libfoo", |
| srcs: ["foo.rs"], |
| crate_name: "foo", |
| host_supported: true, |
| }`) |
| |
| fizzBuzzHost := ctx.ModuleForTests("fizz-buzz", "linux_glibc_x86_64").Module().(*Module) |
| fizzBuzzDevice := ctx.ModuleForTests("fizz-buzz", "android_arm64_armv8-a").Module().(*Module) |
| |
| if !android.InList("libfoo.rlib-std", fizzBuzzHost.Properties.AndroidMkRlibs) { |
| t.Errorf("rustlibs dependency libfoo should be an rlib dep for host modules") |
| } |
| |
| if !android.InList("libfoo", fizzBuzzDevice.Properties.AndroidMkDylibs) { |
| t.Errorf("rustlibs dependency libfoo should be an dylib dep for device modules") |
| } |
| |
| rlibLinkDevice := ctx.ModuleForTests("rlib_linked", "android_arm64_armv8-a").Module().(*Module) |
| |
| if !android.InList("libfoo.rlib-std", rlibLinkDevice.Properties.AndroidMkRlibs) { |
| t.Errorf("rustlibs dependency libfoo should be an rlib dep for device modules when prefer_rlib is set") |
| } |
| } |
| |
| // Test that prefer_rlib links in libstd statically as well as rustlibs. |
| func TestBinaryPreferRlib(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary { |
| name: "rlib_linked", |
| srcs: ["foo.rs"], |
| rustlibs: ["libfoo"], |
| host_supported: true, |
| prefer_rlib: true, |
| } |
| rust_library { |
| name: "libfoo", |
| srcs: ["foo.rs"], |
| crate_name: "foo", |
| host_supported: true, |
| }`) |
| |
| mod := ctx.ModuleForTests("rlib_linked", "android_arm64_armv8-a").Module().(*Module) |
| |
| if !android.InList("libfoo.rlib-std", mod.Properties.AndroidMkRlibs) { |
| t.Errorf("rustlibs dependency libfoo should be an rlib dep when prefer_rlib is defined") |
| } |
| |
| if !android.InList("libstd", mod.Properties.AndroidMkRlibs) { |
| t.Errorf("libstd dependency should be an rlib dep when prefer_rlib is defined") |
| } |
| } |
| |
| // Test that the path returned by HostToolPath is correct |
| func TestHostToolPath(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary_host { |
| name: "fizz-buzz", |
| srcs: ["foo.rs"], |
| }`) |
| |
| path := ctx.ModuleForTests("fizz-buzz", "linux_glibc_x86_64").Module().(*Module).HostToolPath() |
| if g, w := path.String(), "/host/linux-x86/bin/fizz-buzz"; !strings.Contains(g, w) { |
| t.Errorf("wrong host tool path, expected %q got %q", w, g) |
| } |
| } |
| |
| // Test that the flags being passed to rust_binary modules are as expected |
| func TestBinaryFlags(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary_host { |
| name: "fizz-buzz", |
| srcs: ["foo.rs"], |
| }`) |
| |
| fizzBuzz := ctx.ModuleForTests("fizz-buzz", "linux_glibc_x86_64").Rule("rustc") |
| |
| flags := fizzBuzz.Args["rustcFlags"] |
| if strings.Contains(flags, "--test") { |
| t.Errorf("extra --test flag, rustcFlags: %#v", flags) |
| } |
| } |
| |
| // Test that the bootstrap property sets the appropriate linker |
| func TestBootstrap(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary { |
| name: "foo", |
| srcs: ["foo.rs"], |
| bootstrap: true, |
| }`) |
| |
| foo := ctx.ModuleForTests("foo", "android_arm64_armv8-a").Rule("rustc") |
| |
| flag := "-Wl,-dynamic-linker,/system/bin/bootstrap/linker64" |
| if !strings.Contains(foo.Args["linkFlags"], flag) { |
| t.Errorf("missing link flag to use bootstrap linker, expecting %#v, linkFlags: %#v", flag, foo.Args["linkFlags"]) |
| } |
| } |
| |
| func TestStaticBinaryFlags(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary { |
| name: "fizz", |
| srcs: ["foo.rs"], |
| static_executable: true, |
| }`) |
| |
| fizzOut := ctx.ModuleForTests("fizz", "android_arm64_armv8-a").Rule("rustc") |
| fizzMod := ctx.ModuleForTests("fizz", "android_arm64_armv8-a").Module().(*Module) |
| |
| flags := fizzOut.Args["rustcFlags"] |
| linkFlags := fizzOut.Args["linkFlags"] |
| if !strings.Contains(flags, "-C relocation-model=static") { |
| t.Errorf("static binary missing '-C relocation-model=static' in rustcFlags, found: %#v", flags) |
| } |
| if !strings.Contains(flags, "-C panic=abort") { |
| t.Errorf("static binary missing '-C panic=abort' in rustcFlags, found: %#v", flags) |
| } |
| if !strings.Contains(linkFlags, "-static") { |
| t.Errorf("static binary missing '-static' in linkFlags, found: %#v", flags) |
| } |
| |
| if !android.InList("libc", fizzMod.Properties.AndroidMkStaticLibs) { |
| t.Errorf("static binary not linking against libc as a static library") |
| } |
| if len(fizzMod.transitiveAndroidMkSharedLibs.ToList()) > 0 { |
| t.Errorf("static binary incorrectly linking against shared libraries") |
| } |
| } |
| |
| func TestLinkObjects(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary { |
| name: "fizz-buzz", |
| srcs: ["foo.rs"], |
| shared_libs: ["libfoo"], |
| } |
| cc_library { |
| name: "libfoo", |
| }`) |
| |
| fizzBuzz := ctx.ModuleForTests("fizz-buzz", "android_arm64_armv8-a").Rule("rustc") |
| linkFlags := fizzBuzz.Args["linkFlags"] |
| if !strings.Contains(linkFlags, "/libfoo.so") { |
| t.Errorf("missing shared dependency 'libfoo.so' in linkFlags: %#v", linkFlags) |
| } |
| } |
| |
| // Test that stripped versions are correctly generated and used. |
| func TestStrippedBinary(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary { |
| name: "foo", |
| srcs: ["foo.rs"], |
| } |
| rust_binary { |
| name: "bar", |
| srcs: ["foo.rs"], |
| strip: { |
| none: true |
| } |
| } |
| `) |
| |
| foo := ctx.ModuleForTests("foo", "android_arm64_armv8-a") |
| foo.Output("unstripped/foo") |
| foo.Output("foo") |
| |
| // Check that the `cp` rules is using the stripped version as input. |
| cp := foo.Rule("android.Cp") |
| if strings.HasSuffix(cp.Input.String(), "unstripped/foo") { |
| t.Errorf("installed binary not based on stripped version: %v", cp.Input) |
| } |
| |
| fizzBar := ctx.ModuleForTests("bar", "android_arm64_armv8-a").MaybeOutput("unstripped/bar") |
| if fizzBar.Rule != nil { |
| t.Errorf("unstripped binary exists, so stripped binary has incorrectly been generated") |
| } |
| } |