| // 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") | 
 | 	} | 
 | } |