|  | // 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("rustLink") | 
|  |  | 
|  | 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") | 
|  | fizzOutLink := ctx.ModuleForTests("fizz", "android_arm64_armv8-a").Rule("rustLink") | 
|  | fizzMod := ctx.ModuleForTests("fizz", "android_arm64_armv8-a").Module().(*Module) | 
|  |  | 
|  | flags := fizzOut.Args["rustcFlags"] | 
|  | linkFlags := fizzOutLink.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.Properties.AndroidMkSharedLibs) > 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("rustLink") | 
|  | 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") | 
|  | } | 
|  | } |