| // 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 ( |
| "fmt" |
| "os" |
| "runtime" |
| "strings" |
| "testing" |
| |
| "github.com/google/blueprint/proptools" |
| "google.golang.org/protobuf/encoding/prototext" |
| |
| "android/soong/android" |
| "android/soong/cmd/sbox/sbox_proto" |
| "android/soong/genrule" |
| ) |
| |
| func TestMain(m *testing.M) { |
| os.Exit(m.Run()) |
| } |
| |
| var prepareForRustTest = android.GroupFixturePreparers( |
| android.PrepareForTestWithArchMutator, |
| android.PrepareForTestWithDefaults, |
| android.PrepareForTestWithPrebuilts, |
| |
| genrule.PrepareForTestWithGenRuleBuildComponents, |
| |
| PrepareForTestWithRustIncludeVndk, |
| android.FixtureModifyProductVariables(func(variables android.FixtureProductVariables) { |
| variables.DeviceVndkVersion = StringPtr("current") |
| variables.ProductVndkVersion = StringPtr("current") |
| variables.Platform_vndk_version = StringPtr("29") |
| }), |
| ) |
| |
| var rustMockedFiles = android.MockFS{ |
| "foo.rs": nil, |
| "foo.c": nil, |
| "src/bar.rs": nil, |
| "src/any.h": nil, |
| "c_includes/c_header.h": nil, |
| "rust_includes/rust_headers.h": nil, |
| "proto.proto": nil, |
| "proto/buf.proto": nil, |
| "buf.proto": nil, |
| "foo.proto": nil, |
| "liby.so": nil, |
| "libz.so": nil, |
| "data.txt": nil, |
| "liblog.map.txt": nil, |
| } |
| |
| // testRust returns a TestContext in which a basic environment has been setup. |
| // This environment contains a few mocked files. See rustMockedFiles for the list of these files. |
| func testRust(t *testing.T, bp string, preparers ...android.FixturePreparer) *android.TestContext { |
| skipTestIfOsNotSupported(t) |
| result := android.GroupFixturePreparers( |
| prepareForRustTest, |
| rustMockedFiles.AddToFixture(), |
| android.GroupFixturePreparers( |
| preparers..., |
| ), |
| ). |
| RunTestWithBp(t, bp) |
| return result.TestContext |
| } |
| |
| func testRustVndk(t *testing.T, bp string) *android.TestContext { |
| return testRustVndkFs(t, bp, rustMockedFiles) |
| } |
| |
| const ( |
| sharedVendorVariant = "android_vendor.29_arm64_armv8-a_shared" |
| rlibVendorVariant = "android_vendor.29_arm64_armv8-a_rlib_rlib-std" |
| rlibDylibStdVendorVariant = "android_vendor.29_arm64_armv8-a_rlib_rlib-std" |
| dylibVendorVariant = "android_vendor.29_arm64_armv8-a_dylib" |
| sharedRecoveryVariant = "android_recovery_arm64_armv8-a_shared" |
| rlibRecoveryVariant = "android_recovery_arm64_armv8-a_rlib_dylib-std" |
| rlibRlibStdRecoveryVariant = "android_recovery_arm64_armv8-a_rlib_rlib-std" |
| dylibRecoveryVariant = "android_recovery_arm64_armv8-a_dylib" |
| binaryCoreVariant = "android_arm64_armv8-a" |
| binaryVendorVariant = "android_vendor.29_arm64_armv8-a" |
| binaryProductVariant = "android_product.29_arm64_armv8-a" |
| binaryRecoveryVariant = "android_recovery_arm64_armv8-a" |
| ) |
| |
| func testRustVndkFs(t *testing.T, bp string, fs android.MockFS) *android.TestContext { |
| return testRustVndkFsVersions(t, bp, fs, "current", "current", "29") |
| } |
| |
| func testRustVndkFsVersions(t *testing.T, bp string, fs android.MockFS, device_version, product_version, vndk_version string) *android.TestContext { |
| skipTestIfOsNotSupported(t) |
| result := android.GroupFixturePreparers( |
| prepareForRustTest, |
| fs.AddToFixture(), |
| android.FixtureModifyProductVariables( |
| func(variables android.FixtureProductVariables) { |
| variables.DeviceVndkVersion = StringPtr(device_version) |
| variables.ProductVndkVersion = StringPtr(product_version) |
| variables.Platform_vndk_version = StringPtr(vndk_version) |
| }, |
| ), |
| ).RunTestWithBp(t, bp) |
| return result.TestContext |
| } |
| |
| func testRustRecoveryFsVersions(t *testing.T, bp string, fs android.MockFS, device_version, vndk_version, recovery_version string) *android.TestContext { |
| skipTestIfOsNotSupported(t) |
| result := android.GroupFixturePreparers( |
| prepareForRustTest, |
| fs.AddToFixture(), |
| android.FixtureModifyProductVariables( |
| func(variables android.FixtureProductVariables) { |
| variables.DeviceVndkVersion = StringPtr(device_version) |
| variables.RecoverySnapshotVersion = StringPtr(recovery_version) |
| variables.Platform_vndk_version = StringPtr(vndk_version) |
| }, |
| ), |
| ).RunTestWithBp(t, bp) |
| return result.TestContext |
| } |
| |
| // testRustCov returns a TestContext in which a basic environment has been |
| // setup. This environment explicitly enables coverage. |
| func testRustCov(t *testing.T, bp string) *android.TestContext { |
| skipTestIfOsNotSupported(t) |
| result := android.GroupFixturePreparers( |
| prepareForRustTest, |
| rustMockedFiles.AddToFixture(), |
| android.FixtureModifyProductVariables( |
| func(variables android.FixtureProductVariables) { |
| variables.ClangCoverage = proptools.BoolPtr(true) |
| variables.Native_coverage = proptools.BoolPtr(true) |
| variables.NativeCoveragePaths = []string{"*"} |
| }, |
| ), |
| ).RunTestWithBp(t, bp) |
| return result.TestContext |
| } |
| |
| // testRustError ensures that at least one error was raised and its value |
| // matches the pattern provided. The error can be either in the parsing of the |
| // Blueprint or when generating the build actions. |
| func testRustError(t *testing.T, pattern string, bp string) { |
| skipTestIfOsNotSupported(t) |
| android.GroupFixturePreparers( |
| prepareForRustTest, |
| rustMockedFiles.AddToFixture(), |
| ). |
| ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(pattern)). |
| RunTestWithBp(t, bp) |
| } |
| |
| // testRustVndkError is similar to testRustError, but can be used to test VNDK-related errors. |
| func testRustVndkError(t *testing.T, pattern string, bp string) { |
| testRustVndkFsError(t, pattern, bp, rustMockedFiles) |
| } |
| |
| func testRustVndkFsError(t *testing.T, pattern string, bp string, fs android.MockFS) { |
| skipTestIfOsNotSupported(t) |
| android.GroupFixturePreparers( |
| prepareForRustTest, |
| fs.AddToFixture(), |
| android.FixtureModifyProductVariables( |
| func(variables android.FixtureProductVariables) { |
| variables.DeviceVndkVersion = StringPtr("current") |
| variables.ProductVndkVersion = StringPtr("current") |
| variables.Platform_vndk_version = StringPtr("VER") |
| }, |
| ), |
| ). |
| ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(pattern)). |
| RunTestWithBp(t, bp) |
| } |
| |
| // testRustCtx is used to build a particular test environment. Unless your |
| // tests requires a specific setup, prefer the wrapping functions: testRust, |
| // testRustCov or testRustError. |
| type testRustCtx struct { |
| bp string |
| fs map[string][]byte |
| env map[string]string |
| config *android.Config |
| } |
| |
| func skipTestIfOsNotSupported(t *testing.T) { |
| // TODO (b/140435149) |
| if runtime.GOOS != "linux" { |
| t.Skip("Rust Soong tests can only be run on Linux hosts currently") |
| } |
| } |
| |
| // Test that we can extract the link path from a lib path. |
| func TestLinkPathFromFilePath(t *testing.T) { |
| barPath := android.PathForTesting("out/soong/.intermediates/external/libbar/libbar/linux_glibc_x86_64_shared/libbar.so") |
| libName := barPath.Dir() |
| expectedResult := "out/soong/.intermediates/external/libbar/libbar/linux_glibc_x86_64_shared" |
| |
| if libName.String() != expectedResult { |
| t.Errorf("libNameFromFilePath returned the wrong name; expected '%#v', got '%#v'", expectedResult, libName.String()) |
| } |
| } |
| |
| // Test to make sure dependencies are being picked up correctly. |
| func TestDepsTracking(t *testing.T) { |
| ctx := testRust(t, ` |
| cc_library { |
| host_supported: true, |
| name: "cc_stubs_dep", |
| } |
| rust_ffi_host_static { |
| name: "libstatic", |
| srcs: ["foo.rs"], |
| crate_name: "static", |
| } |
| rust_ffi_host_static { |
| name: "libwholestatic", |
| srcs: ["foo.rs"], |
| crate_name: "wholestatic", |
| } |
| rust_ffi_host_shared { |
| name: "libshared", |
| srcs: ["foo.rs"], |
| crate_name: "shared", |
| } |
| rust_library_host_rlib { |
| name: "librlib", |
| srcs: ["foo.rs"], |
| crate_name: "rlib", |
| static_libs: ["libstatic"], |
| whole_static_libs: ["libwholestatic"], |
| shared_libs: ["cc_stubs_dep"], |
| } |
| rust_proc_macro { |
| name: "libpm", |
| srcs: ["foo.rs"], |
| crate_name: "pm", |
| } |
| rust_binary_host { |
| name: "fizz-buzz", |
| rlibs: ["librlib"], |
| proc_macros: ["libpm"], |
| static_libs: ["libstatic"], |
| shared_libs: ["libshared"], |
| srcs: ["foo.rs"], |
| } |
| `) |
| module := ctx.ModuleForTests("fizz-buzz", "linux_glibc_x86_64").Module().(*Module) |
| rustc := ctx.ModuleForTests("librlib", "linux_glibc_x86_64_rlib_rlib-std").Rule("rustc") |
| rustLink := ctx.ModuleForTests("fizz-buzz", "linux_glibc_x86_64").Rule("rustc") |
| |
| // Since dependencies are added to AndroidMk* properties, we can check these to see if they've been picked up. |
| if !android.InList("librlib.rlib-std", module.Properties.AndroidMkRlibs) { |
| t.Errorf("Rlib dependency not detected (dependency missing from AndroidMkRlibs)") |
| } |
| |
| if !android.InList("libpm", module.Properties.AndroidMkProcMacroLibs) { |
| t.Errorf("Proc_macro dependency not detected (dependency missing from AndroidMkProcMacroLibs)") |
| } |
| |
| if !android.InList("libshared", module.transitiveAndroidMkSharedLibs.ToList()) { |
| t.Errorf("Shared library dependency not detected (dependency missing from AndroidMkSharedLibs)") |
| } |
| |
| if !android.InList("libstatic", module.Properties.AndroidMkStaticLibs) { |
| t.Errorf("Static library dependency not detected (dependency missing from AndroidMkStaticLibs)") |
| } |
| |
| if !strings.Contains(rustc.RuleParams.Command, "-lstatic=wholestatic") { |
| t.Errorf("-lstatic flag not being passed to rustc for static library %#v", rustc.RuleParams.Command) |
| } |
| |
| if !strings.Contains(rustLink.RuleParams.Command, "cc_stubs_dep.so") { |
| t.Errorf("shared cc_library not being passed to rustc linkFlags %#v", rustLink.RuleParams.Command) |
| } |
| |
| if !android.SuffixInList(rustLink.Implicits.Strings(), "cc_stubs_dep.so") { |
| t.Errorf("shared cc dep not being passed as implicit to rustc %#v", rustLink.OrderOnly.Strings()) |
| } |
| |
| if !android.SuffixInList(rustLink.Implicits.Strings(), "cc_stubs_dep.so.toc") { |
| t.Errorf("shared cc dep TOC not being passed as implicit to rustc %#v", rustLink.Implicits.Strings()) |
| } |
| } |
| |
| func TestSourceProviderDeps(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary { |
| name: "fizz-buzz-dep", |
| srcs: [ |
| "foo.rs", |
| ":my_generator", |
| ":libbindings", |
| ], |
| rlibs: ["libbindings"], |
| } |
| rust_proc_macro { |
| name: "libprocmacro", |
| srcs: [ |
| "foo.rs", |
| ":my_generator", |
| ":libbindings", |
| ], |
| rlibs: ["libbindings"], |
| crate_name: "procmacro", |
| } |
| rust_library { |
| name: "libfoo", |
| srcs: [ |
| "foo.rs", |
| ":my_generator", |
| ":libbindings", |
| ], |
| rlibs: ["libbindings"], |
| crate_name: "foo", |
| } |
| genrule { |
| name: "my_generator", |
| tools: ["any_rust_binary"], |
| cmd: "$(location) -o $(out) $(in)", |
| srcs: ["src/any.h"], |
| out: ["src/any.rs"], |
| } |
| rust_binary_host { |
| name: "any_rust_binary", |
| srcs: [ |
| "foo.rs", |
| ], |
| } |
| rust_bindgen { |
| name: "libbindings", |
| crate_name: "bindings", |
| source_stem: "bindings", |
| host_supported: true, |
| wrapper_src: "src/any.h", |
| } |
| `) |
| |
| libfoo := ctx.ModuleForTests("libfoo", "android_arm64_armv8-a_rlib_dylib-std").Rule("rustc") |
| if !android.SuffixInList(libfoo.Implicits.Strings(), "/out/bindings.rs") { |
| t.Errorf("rust_bindgen generated source not included as implicit input for libfoo; Implicits %#v", libfoo.Implicits.Strings()) |
| } |
| if !android.SuffixInList(libfoo.Implicits.Strings(), "/out/any.rs") { |
| t.Errorf("genrule generated source not included as implicit input for libfoo; Implicits %#v", libfoo.Implicits.Strings()) |
| } |
| |
| fizzBuzz := ctx.ModuleForTests("fizz-buzz-dep", "android_arm64_armv8-a").Rule("rustc") |
| if !android.SuffixInList(fizzBuzz.Implicits.Strings(), "/out/bindings.rs") { |
| t.Errorf("rust_bindgen generated source not included as implicit input for fizz-buzz-dep; Implicits %#v", libfoo.Implicits.Strings()) |
| } |
| if !android.SuffixInList(fizzBuzz.Implicits.Strings(), "/out/any.rs") { |
| t.Errorf("genrule generated source not included as implicit input for fizz-buzz-dep; Implicits %#v", libfoo.Implicits.Strings()) |
| } |
| |
| libprocmacro := ctx.ModuleForTests("libprocmacro", "linux_glibc_x86_64").Rule("rustc") |
| if !android.SuffixInList(libprocmacro.Implicits.Strings(), "/out/bindings.rs") { |
| t.Errorf("rust_bindgen generated source not included as implicit input for libprocmacro; Implicits %#v", libfoo.Implicits.Strings()) |
| } |
| if !android.SuffixInList(libprocmacro.Implicits.Strings(), "/out/any.rs") { |
| t.Errorf("genrule generated source not included as implicit input for libprocmacro; Implicits %#v", libfoo.Implicits.Strings()) |
| } |
| |
| // Check that our bindings are picked up as crate dependencies as well |
| libfooMod := ctx.ModuleForTests("libfoo", "android_arm64_armv8-a_dylib").Module().(*Module) |
| if !android.InList("libbindings", libfooMod.Properties.AndroidMkRlibs) { |
| t.Errorf("bindgen dependency not detected as a rlib dependency (dependency missing from AndroidMkRlibs)") |
| } |
| fizzBuzzMod := ctx.ModuleForTests("fizz-buzz-dep", "android_arm64_armv8-a").Module().(*Module) |
| if !android.InList("libbindings", fizzBuzzMod.Properties.AndroidMkRlibs) { |
| t.Errorf("bindgen dependency not detected as a rlib dependency (dependency missing from AndroidMkRlibs)") |
| } |
| libprocmacroMod := ctx.ModuleForTests("libprocmacro", "linux_glibc_x86_64").Module().(*Module) |
| if !android.InList("libbindings.rlib-std", libprocmacroMod.Properties.AndroidMkRlibs) { |
| t.Errorf("bindgen dependency not detected as a rlib dependency (dependency missing from AndroidMkRlibs)") |
| } |
| } |
| |
| func TestSourceProviderTargetMismatch(t *testing.T) { |
| // This might error while building the dependency tree or when calling depsToPaths() depending on the lunched |
| // target, which results in two different errors. So don't check the error, just confirm there is one. |
| testRustError(t, ".*", ` |
| rust_proc_macro { |
| name: "libprocmacro", |
| srcs: [ |
| "foo.rs", |
| ":libbindings", |
| ], |
| crate_name: "procmacro", |
| } |
| rust_bindgen { |
| name: "libbindings", |
| crate_name: "bindings", |
| source_stem: "bindings", |
| wrapper_src: "src/any.h", |
| } |
| `) |
| } |
| |
| // Test to make sure proc_macros use host variants when building device modules. |
| func TestProcMacroDeviceDeps(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_library_host_rlib { |
| name: "libbar", |
| srcs: ["foo.rs"], |
| crate_name: "bar", |
| } |
| rust_proc_macro { |
| name: "libpm", |
| rlibs: ["libbar"], |
| srcs: ["foo.rs"], |
| crate_name: "pm", |
| } |
| rust_binary { |
| name: "fizz-buzz", |
| proc_macros: ["libpm"], |
| srcs: ["foo.rs"], |
| } |
| `) |
| rustc := ctx.ModuleForTests("libpm", "linux_glibc_x86_64").Rule("rustc") |
| |
| if !strings.Contains(rustc.RuleParams.Command, "libbar/linux_glibc_x86_64") { |
| t.Errorf("Proc_macro is not using host variant of dependent modules.") |
| } |
| } |
| |
| // Test that no_stdlibs suppresses dependencies on rust standard libraries |
| func TestNoStdlibs(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_binary { |
| name: "fizz-buzz", |
| srcs: ["foo.rs"], |
| no_stdlibs: true, |
| }`) |
| module := ctx.ModuleForTests("fizz-buzz", "android_arm64_armv8-a").Module().(*Module) |
| |
| if android.InList("libstd", module.Properties.AndroidMkDylibs) { |
| t.Errorf("no_stdlibs did not suppress dependency on libstd") |
| } |
| } |
| |
| // Test that libraries provide both 32-bit and 64-bit variants. |
| func TestMultilib(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_library_rlib { |
| name: "libfoo", |
| srcs: ["foo.rs"], |
| crate_name: "foo", |
| }`) |
| |
| _ = ctx.ModuleForTests("libfoo", "android_arm64_armv8-a_rlib_dylib-std") |
| _ = ctx.ModuleForTests("libfoo", "android_arm_armv7-a-neon_rlib_dylib-std") |
| } |
| |
| // Test that library size measurements are generated. |
| func TestLibrarySizes(t *testing.T) { |
| ctx := testRust(t, ` |
| rust_library_dylib { |
| name: "libwaldo", |
| srcs: ["foo.rs"], |
| crate_name: "waldo", |
| }`) |
| |
| m := ctx.SingletonForTests("file_metrics") |
| m.Output("unstripped/libwaldo.dylib.so.bloaty.csv") |
| m.Output("libwaldo.dylib.so.bloaty.csv") |
| } |
| |
| func assertString(t *testing.T, got, expected string) { |
| t.Helper() |
| if got != expected { |
| t.Errorf("expected %q got %q", expected, got) |
| } |
| } |
| |
| var ( |
| sboxCompilationFiles = []string{ |
| "out/soong/.intermediates/defaults/rust/libaddr2line/android_arm64_armv8-a_rlib/libaddr2line.rlib", |
| "out/soong/.intermediates/defaults/rust/libadler/android_arm64_armv8-a_rlib/libadler.rlib", |
| "out/soong/.intermediates/defaults/rust/liballoc/android_arm64_armv8-a_rlib/liballoc.rlib", |
| "out/soong/.intermediates/defaults/rust/libcfg_if/android_arm64_armv8-a_rlib/libcfg_if.rlib", |
| "out/soong/.intermediates/defaults/rust/libcompiler_builtins/android_arm64_armv8-a_rlib/libcompiler_builtins.rlib", |
| "out/soong/.intermediates/defaults/rust/libcore/android_arm64_armv8-a_rlib/libcore.rlib", |
| "out/soong/.intermediates/defaults/rust/libgimli/android_arm64_armv8-a_rlib/libgimli.rlib", |
| "out/soong/.intermediates/defaults/rust/libhashbrown/android_arm64_armv8-a_rlib/libhashbrown.rlib", |
| "out/soong/.intermediates/defaults/rust/liblibc/android_arm64_armv8-a_rlib/liblibc.rlib", |
| "out/soong/.intermediates/defaults/rust/libmemchr/android_arm64_armv8-a_rlib/libmemchr.rlib", |
| "out/soong/.intermediates/defaults/rust/libminiz_oxide/android_arm64_armv8-a_rlib/libminiz_oxide.rlib", |
| "out/soong/.intermediates/defaults/rust/libobject/android_arm64_armv8-a_rlib/libobject.rlib", |
| "out/soong/.intermediates/defaults/rust/libpanic_unwind/android_arm64_armv8-a_rlib/libpanic_unwind.rlib", |
| "out/soong/.intermediates/defaults/rust/librustc_demangle/android_arm64_armv8-a_rlib/librustc_demangle.rlib", |
| "out/soong/.intermediates/defaults/rust/librustc_std_workspace_alloc/android_arm64_armv8-a_rlib/librustc_std_workspace_alloc.rlib", |
| "out/soong/.intermediates/defaults/rust/librustc_std_workspace_core/android_arm64_armv8-a_rlib/librustc_std_workspace_core.rlib", |
| "out/soong/.intermediates/defaults/rust/libstd_detect/android_arm64_armv8-a_rlib/libstd_detect.rlib", |
| "build/soong/scripts/mkcratersp.py", |
| "defaults/rust/linux-x86/1.69.0/bin/rustc", |
| "defaults/rust/linux-x86/1.69.0/lib/libstd.so", |
| "defaults/rust/linux-x86/1.69.0/lib64/libc++.so.1", |
| } |
| sboxCompilationFilesWithCc = []string{ |
| "defaults/cc/common", |
| "out/soong/.intermediates/defaults/cc/common/libc/android_arm64_armv8-a_shared/libc.so", |
| "out/soong/.intermediates/defaults/cc/common/libc/android_arm64_armv8-a_shared/libc.so.toc", |
| "out/soong/.intermediates/defaults/cc/common/libdl/android_arm64_armv8-a_shared/libdl.so", |
| "out/soong/.intermediates/defaults/cc/common/libdl/android_arm64_armv8-a_shared/libdl.so.toc", |
| "out/soong/.intermediates/defaults/cc/common/libm/android_arm64_armv8-a_shared/libm.so", |
| "out/soong/.intermediates/defaults/cc/common/libm/android_arm64_armv8-a_shared/libm.so.toc", |
| "out/soong/.intermediates/defaults/rust/liblog/android_arm64_armv8-a_shared/liblog.so", |
| "out/soong/.intermediates/defaults/rust/liblog/android_arm64_armv8-a_shared/liblog.so.toc", |
| } |
| ) |
| |
| func TestSandboxCompilation(t *testing.T) { |
| ctx := testRust(t, ` |
| filegroup { |
| name: "libsrcs1", |
| srcs: ["src_filegroup1.rs"], |
| } |
| filegroup { |
| name: "libsrcs2", |
| srcs: ["src_filegroup2.rs"], |
| } |
| rust_library { |
| name: "libfizz_buzz", |
| crate_name:"fizz_buzz", |
| crate_root: "foo.rs", |
| srcs: [ |
| "src_lib*.rs", |
| ":libsrcs1", |
| ":libsrcs2", |
| ], |
| compile_data: [ |
| "compile_data1.txt", |
| "compile_data2.txt", |
| ], |
| dylib: { |
| srcs: ["dylib_only.rs"], |
| }, |
| rlib: { |
| srcs: ["rlib_only.rs"], |
| }, |
| } |
| rust_binary { |
| name: "fizz_buzz", |
| crate_name:"fizz_buzz", |
| crate_root: "foo.rs", |
| srcs: [ |
| "src_lib*.rs", |
| ":libsrcs1", |
| ":libsrcs2", |
| ], |
| } |
| rust_ffi { |
| name: "librust_ffi", |
| crate_name: "rust_ffi", |
| crate_root: "foo.rs", |
| static: { |
| srcs: ["static_only.rs"], |
| }, |
| shared: { |
| srcs: ["shared_only.rs"], |
| }, |
| srcs: ["src1.rs"], |
| } |
| cc_library_static { |
| name: "cc_dep_static", |
| } |
| cc_library_shared { |
| name: "cc_dep_shared", |
| } |
| rust_library { |
| name: "libfizz_buzz_cc_deps", |
| crate_name:"fizz_buzz", |
| crate_root: "foo.rs", |
| srcs: ["src*.rs"], |
| shared_libs: ["cc_dep_shared"], |
| static_libs: ["cc_dep_static"], |
| } |
| rust_library { |
| name: "libfizz_buzz_intermediate_cc_deps", |
| crate_name:"fizz_buzz", |
| crate_root: "foo.rs", |
| srcs: ["src*.rs"], |
| rustlibs: ["libfizz_buzz_cc_deps"], |
| } |
| rust_library { |
| name: "libfizz_buzz_transitive_cc_deps", |
| crate_name:"fizz_buzz", |
| crate_root: "foo.rs", |
| srcs: ["src*.rs"], |
| rustlibs: ["libfizz_buzz_intermediate_cc_deps"], |
| } |
| `, |
| android.FixtureMergeMockFs(android.MockFS{ |
| "src_lib1.rs": nil, |
| "src_lib2.rs": nil, |
| "src_lib3.rs": nil, |
| "src_lib4.rs": nil, |
| "src_filegroup1.rs": nil, |
| "src_filegroup2.rs": nil, |
| "static_only.rs": nil, |
| "shared_only.rs": nil, |
| }), |
| ) |
| |
| testcases := []struct { |
| name string |
| moduleName string |
| variant string |
| rustcExpectedFilesToCopy []string |
| expectedFlags []string |
| }{ |
| { |
| name: "rust_library (dylib)", |
| moduleName: "libfizz_buzz", |
| variant: "android_arm64_armv8-a_dylib", |
| rustcExpectedFilesToCopy: android.Concat(sboxCompilationFiles, sboxCompilationFilesWithCc, []string{ |
| "foo.rs", |
| "src_lib1.rs", |
| "src_lib2.rs", |
| "src_lib3.rs", |
| "src_lib4.rs", |
| "src_filegroup1.rs", |
| "src_filegroup2.rs", |
| "compile_data1.txt", |
| "compile_data2.txt", |
| "dylib_only.rs", |
| "out/soong/.intermediates/libfizz_buzz/android_arm64_armv8-a_dylib/out/src_filegroup1.rs", |
| "out/soong/.intermediates/libfizz_buzz/android_arm64_armv8-a_dylib/out/src_filegroup2.rs", |
| |
| "out/soong/.intermediates/defaults/cc/common/libc/android_arm64_armv8-a_shared/libc.so", |
| "out/soong/.intermediates/defaults/cc/common/libc/android_arm64_armv8-a_shared/libc.so.toc", |
| "out/soong/.intermediates/defaults/cc/common/libm/android_arm64_armv8-a_shared/libm.so", |
| "out/soong/.intermediates/defaults/cc/common/libm/android_arm64_armv8-a_shared/libm.so.toc", |
| "out/soong/.intermediates/defaults/cc/common/libdl/android_arm64_armv8-a_shared/libdl.so", |
| "out/soong/.intermediates/defaults/cc/common/libdl/android_arm64_armv8-a_shared/libdl.so.toc", |
| "out/soong/.intermediates/defaults/rust/libstd/android_arm64_armv8-a_dylib/unstripped/libstd.dylib.so", |
| "out/soong/.intermediates/defaults/cc/common/crtbegin_so/android_arm64_armv8-a/crtbegin_so.o", |
| "out/soong/.intermediates/defaults/cc/common/crtend_so/android_arm64_armv8-a/crtend_so.o", |
| "out/soong/.intermediates/libfizz_buzz/android_arm64_armv8-a_dylib/libfizz_buzz.dylib.so.clippy", |
| }), |
| expectedFlags: []string{ |
| "-C linker=build/soong/scripts/mkcratersp.py", |
| "--emit link", |
| "-o __SBOX_SANDBOX_DIR__/out/libfizz_buzz.dylib.so.rsp", |
| "--emit dep-info=__SBOX_SANDBOX_DIR__/out/libfizz_buzz.dylib.so.d.raw", |
| "foo.rs", // this is the entry point |
| }, |
| }, |
| { |
| name: "rust_library (rlib dylib-std)", |
| moduleName: "libfizz_buzz", |
| variant: "android_arm64_armv8-a_rlib_dylib-std", |
| rustcExpectedFilesToCopy: android.Concat(sboxCompilationFiles, []string{ |
| "foo.rs", |
| "src_lib1.rs", |
| "src_lib2.rs", |
| "src_lib3.rs", |
| "src_lib4.rs", |
| "src_filegroup1.rs", |
| "src_filegroup2.rs", |
| "rlib_only.rs", |
| "out/soong/.intermediates/libfizz_buzz/android_arm64_armv8-a_rlib_dylib-std/out/src_filegroup1.rs", |
| "out/soong/.intermediates/libfizz_buzz/android_arm64_armv8-a_rlib_dylib-std/out/src_filegroup2.rs", |
| "out/soong/.intermediates/defaults/rust/libstd/android_arm64_armv8-a_dylib/unstripped/libstd.dylib.so", |
| }), |
| expectedFlags: []string{ |
| "--emit link", |
| "-o __SBOX_SANDBOX_DIR__/out/libfizz_buzz.rlib", |
| "--emit dep-info=__SBOX_SANDBOX_DIR__/out/libfizz_buzz.rlib.d.raw", |
| "foo.rs", // this is the entry point |
| }, |
| }, |
| { |
| name: "rust_library (rlib rlib-std)", |
| moduleName: "libfizz_buzz", |
| variant: "android_arm64_armv8-a_rlib_rlib-std", |
| rustcExpectedFilesToCopy: android.Concat(sboxCompilationFiles, []string{ |
| "foo.rs", |
| "src_lib1.rs", |
| "src_lib2.rs", |
| "src_lib3.rs", |
| "src_lib4.rs", |
| "src_filegroup1.rs", |
| "src_filegroup2.rs", |
| "rlib_only.rs", |
| "out/soong/.intermediates/libfizz_buzz/android_arm64_armv8-a_rlib_rlib-std/out/src_filegroup1.rs", |
| "out/soong/.intermediates/libfizz_buzz/android_arm64_armv8-a_rlib_rlib-std/out/src_filegroup2.rs", |
| "out/soong/.intermediates/libfizz_buzz/android_arm64_armv8-a_rlib_rlib-std/libfizz_buzz.rlib.clippy", |
| "out/soong/.intermediates/defaults/rust/libstd/android_arm64_armv8-a_rlib/libstd.rlib", |
| }), |
| expectedFlags: []string{ |
| "--emit link", |
| "-o __SBOX_SANDBOX_DIR__/out/libfizz_buzz.rlib", |
| "--emit dep-info=__SBOX_SANDBOX_DIR__/out/libfizz_buzz.rlib.d.raw", |
| "foo.rs", // this is the entry point |
| }, |
| }, |
| { |
| name: "rust_binary", |
| moduleName: "fizz_buzz", |
| variant: "android_arm64_armv8-a", |
| rustcExpectedFilesToCopy: android.Concat(sboxCompilationFiles, sboxCompilationFilesWithCc, []string{ |
| "foo.rs", |
| "src_lib1.rs", |
| "src_lib2.rs", |
| "src_lib3.rs", |
| "src_lib4.rs", |
| "src_filegroup1.rs", |
| "src_filegroup2.rs", |
| "out/soong/.intermediates/fizz_buzz/android_arm64_armv8-a/out/src_filegroup1.rs", |
| "out/soong/.intermediates/fizz_buzz/android_arm64_armv8-a/out/src_filegroup2.rs", |
| |
| "out/soong/.intermediates/defaults/rust/libstd/android_arm64_armv8-a_dylib/unstripped/libstd.dylib.so", |
| "out/soong/.intermediates/defaults/cc/common/crtbegin_dynamic/android_arm64_armv8-a/crtbegin_dynamic.o", |
| "out/soong/.intermediates/defaults/cc/common/crtend_android/android_arm64_armv8-a/crtend_android.o", |
| "out/soong/.intermediates/fizz_buzz/android_arm64_armv8-a/fizz_buzz.clippy", |
| }), |
| expectedFlags: []string{ |
| "--emit link", |
| "-o __SBOX_SANDBOX_DIR__/out/fizz_buzz", |
| "--emit dep-info=__SBOX_SANDBOX_DIR__/out/fizz_buzz.d.raw", |
| "foo.rs", // this is the entry point |
| }, |
| }, |
| { |
| name: "rust_ffi static lib variant", |
| moduleName: "librust_ffi", |
| variant: "android_arm64_armv8-a_static", |
| rustcExpectedFilesToCopy: android.Concat(sboxCompilationFiles, []string{ |
| "foo.rs", |
| "src1.rs", |
| "static_only.rs", |
| "out/soong/.intermediates/librust_ffi/android_arm64_armv8-a_static/librust_ffi.a.clippy", |
| "out/soong/.intermediates/defaults/rust/libstd/android_arm64_armv8-a_rlib/libstd.rlib", |
| }), |
| expectedFlags: []string{ |
| "--emit link", |
| "-o __SBOX_SANDBOX_DIR__/out/librust_ffi.a", |
| "--emit dep-info=__SBOX_SANDBOX_DIR__/out/librust_ffi.a.d.raw", |
| "foo.rs", // this is the entry point |
| }, |
| }, |
| { |
| name: "rust_ffi shared lib variant", |
| moduleName: "librust_ffi", |
| variant: "android_arm64_armv8-a_shared", |
| rustcExpectedFilesToCopy: android.Concat(sboxCompilationFiles, sboxCompilationFilesWithCc, []string{ |
| "foo.rs", |
| "src1.rs", |
| "shared_only.rs", |
| |
| "out/soong/.intermediates/defaults/rust/libstd/android_arm64_armv8-a_dylib/unstripped/libstd.dylib.so", |
| "out/soong/.intermediates/defaults/cc/common/crtbegin_so/android_arm64_armv8-a/crtbegin_so.o", |
| "out/soong/.intermediates/defaults/cc/common/crtend_so/android_arm64_armv8-a/crtend_so.o", |
| "out/soong/.intermediates/librust_ffi/android_arm64_armv8-a_shared/librust_ffi.so.clippy", |
| }), |
| expectedFlags: []string{ |
| "--emit link", |
| "-o __SBOX_SANDBOX_DIR__/out/librust_ffi.so", |
| "--emit dep-info=__SBOX_SANDBOX_DIR__/out/librust_ffi.so.d.raw", |
| "foo.rs", // this is the entry point |
| }, |
| }, |
| { |
| name: "rust_library with cc deps (dylib)", |
| moduleName: "libfizz_buzz_cc_deps", |
| variant: "android_arm64_armv8-a_dylib", |
| rustcExpectedFilesToCopy: []string{ |
| "out/soong/.intermediates/cc_dep_static/android_arm64_armv8-a_static/cc_dep_static.a", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so.toc", |
| }, |
| }, |
| { |
| name: "rust_library with cc deps (rlib rlib-std)", |
| moduleName: "libfizz_buzz_cc_deps", |
| variant: "android_arm64_armv8-a_rlib_rlib-std", |
| rustcExpectedFilesToCopy: []string{ |
| "out/soong/.intermediates/cc_dep_static/android_arm64_armv8-a_static/cc_dep_static.a", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so.toc", |
| }, |
| }, |
| { |
| name: "rust_library with cc deps (rlib dylib-std)", |
| moduleName: "libfizz_buzz_cc_deps", |
| variant: "android_arm64_armv8-a_rlib_dylib-std", |
| rustcExpectedFilesToCopy: []string{ |
| "out/soong/.intermediates/cc_dep_static/android_arm64_armv8-a_static/cc_dep_static.a", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so.toc", |
| }, |
| }, |
| { |
| name: "rust_library with transitive cc deps (dylib)", |
| moduleName: "libfizz_buzz_transitive_cc_deps", |
| variant: "android_arm64_armv8-a_dylib", |
| rustcExpectedFilesToCopy: []string{ |
| "out/soong/.intermediates/cc_dep_static/android_arm64_armv8-a_static/cc_dep_static.a", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so.toc", |
| }, |
| }, |
| { |
| name: "rust_library with transitive cc deps (rlib rlib-std)", |
| moduleName: "libfizz_buzz_transitive_cc_deps", |
| variant: "android_arm64_armv8-a_rlib_rlib-std", |
| rustcExpectedFilesToCopy: []string{ |
| "out/soong/.intermediates/cc_dep_static/android_arm64_armv8-a_static/cc_dep_static.a", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so.toc", |
| }, |
| }, |
| { |
| name: "rust_library with transitive cc deps (rlib dylib-std)", |
| moduleName: "libfizz_buzz_transitive_cc_deps", |
| variant: "android_arm64_armv8-a_rlib_dylib-std", |
| rustcExpectedFilesToCopy: []string{ |
| "out/soong/.intermediates/cc_dep_static/android_arm64_armv8-a_static/cc_dep_static.a", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so", |
| "out/soong/.intermediates/cc_dep_shared/android_arm64_armv8-a_shared/cc_dep_shared.so.toc", |
| }, |
| }, |
| } |
| |
| for _, tc := range testcases { |
| t.Run(tc.name, func(t *testing.T) { |
| writeFile := ctx.ModuleForTests(tc.moduleName, tc.variant).Rule("unescapedWriteFile") |
| contents := writeFile.BuildParams.Args["content"] |
| manifestProto := sbox_proto.Manifest{} |
| err := prototext.Unmarshal([]byte(contents), &manifestProto) |
| if err != nil { |
| t.Errorf("expected no errors unmarshaling manifest proto; got %v", err) |
| } |
| |
| if len(manifestProto.Commands) != 1 { |
| t.Errorf("expected 1 command; got %v", len(manifestProto.Commands)) |
| } |
| |
| // check that sandbox contains correct files |
| rustc := manifestProto.Commands[0] |
| actualFilesToCopy := []string{} |
| for _, copy := range rustc.CopyBefore { |
| actualFilesToCopy = append(actualFilesToCopy, copy.GetFrom()) |
| } |
| _, expectedFilesNotCopied, _ := android.ListSetDifference(tc.rustcExpectedFilesToCopy, actualFilesToCopy) |
| if len(expectedFilesNotCopied) > 0 { |
| t.Errorf("did not copy expected files to sbox: %v;\n files copied: %v", expectedFilesNotCopied, actualFilesToCopy) |
| } |
| |
| rustcCmd := proptools.String(rustc.Command) |
| for _, flag := range tc.expectedFlags { |
| android.AssertStringDoesContain( |
| t, |
| fmt.Sprintf( |
| "missing flag in rustc invocation; expected to find substring %q; got %q", |
| flag, |
| rustcCmd, |
| ), |
| rustcCmd, |
| flag, |
| ) |
| } |
| }) |
| } |
| } |