| // Copyright 2020 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 coverage flags are being correctly generated. |
| func TestCoverageFlags(t *testing.T) { |
| ctx := testRustCov(t, ` |
| rust_library { |
| name: "libfoo_cov", |
| srcs: ["foo.rs"], |
| crate_name: "foo", |
| } |
| rust_binary { |
| name: "fizz_cov", |
| srcs: ["foo.rs"], |
| } |
| rust_binary { |
| name: "buzzNoCov", |
| srcs: ["foo.rs"], |
| native_coverage: false, |
| } |
| rust_library { |
| name: "libbar_nocov", |
| srcs: ["foo.rs"], |
| crate_name: "bar", |
| native_coverage: false, |
| }`) |
| |
| // Make sure native_coverage: false isn't creating a coverage variant. |
| if android.InList("android_arm64_armv8-a_dylib_cov", ctx.ModuleVariantsForTests("libbar_nocov")) { |
| t.Fatalf("coverage variant created for module 'libbar_nocov' with native coverage disabled") |
| } |
| |
| // Just test the dylib variants unless the library coverage logic changes to distinguish between the types. |
| libfooCov := ctx.ModuleForTests("libfoo_cov", "android_arm64_armv8-a_dylib_cov").Rule("rustc") |
| libbarNoCov := ctx.ModuleForTests("libbar_nocov", "android_arm64_armv8-a_dylib").Rule("rustc") |
| fizzCov := ctx.ModuleForTests("fizz_cov", "android_arm64_armv8-a_cov").Rule("rustc") |
| buzzNoCov := ctx.ModuleForTests("buzzNoCov", "android_arm64_armv8-a").Rule("rustc") |
| |
| rustcCoverageFlags := []string{"-Z profile", " -g ", "-C opt-level=0", "-C link-dead-code", "-Z no-landing-pads"} |
| for _, flag := range rustcCoverageFlags { |
| missingErrorStr := "missing rustc flag '%s' for '%s' module with coverage enabled; rustcFlags: %#v" |
| containsErrorStr := "contains rustc flag '%s' for '%s' module with coverage disabled; rustcFlags: %#v" |
| |
| if !strings.Contains(fizzCov.Args["rustcFlags"], flag) { |
| t.Fatalf(missingErrorStr, flag, "fizz_cov", fizzCov.Args["rustcFlags"]) |
| } |
| if !strings.Contains(libfooCov.Args["rustcFlags"], flag) { |
| t.Fatalf(missingErrorStr, flag, "libfoo_cov dylib", libfooCov.Args["rustcFlags"]) |
| } |
| if strings.Contains(buzzNoCov.Args["rustcFlags"], flag) { |
| t.Fatalf(containsErrorStr, flag, "buzzNoCov", buzzNoCov.Args["rustcFlags"]) |
| } |
| if strings.Contains(libbarNoCov.Args["rustcFlags"], flag) { |
| t.Fatalf(containsErrorStr, flag, "libbar_cov", libbarNoCov.Args["rustcFlags"]) |
| } |
| } |
| |
| linkCoverageFlags := []string{"--coverage", " -g "} |
| for _, flag := range linkCoverageFlags { |
| missingErrorStr := "missing rust linker flag '%s' for '%s' module with coverage enabled; rustcFlags: %#v" |
| containsErrorStr := "contains rust linker flag '%s' for '%s' module with coverage disabled; rustcFlags: %#v" |
| |
| if !strings.Contains(fizzCov.Args["linkFlags"], flag) { |
| t.Fatalf(missingErrorStr, flag, "fizz_cov", fizzCov.Args["linkFlags"]) |
| } |
| if !strings.Contains(libfooCov.Args["linkFlags"], flag) { |
| t.Fatalf(missingErrorStr, flag, "libfoo_cov dylib", libfooCov.Args["linkFlags"]) |
| } |
| if strings.Contains(buzzNoCov.Args["linkFlags"], flag) { |
| t.Fatalf(containsErrorStr, flag, "buzzNoCov", buzzNoCov.Args["linkFlags"]) |
| } |
| if strings.Contains(libbarNoCov.Args["linkFlags"], flag) { |
| t.Fatalf(containsErrorStr, flag, "libbar_cov", libbarNoCov.Args["linkFlags"]) |
| } |
| } |
| |
| } |
| |
| // Test coverage files are included correctly |
| func TestCoverageZip(t *testing.T) { |
| ctx := testRustCov(t, ` |
| rust_library { |
| name: "libfoo", |
| srcs: ["foo.rs"], |
| rlibs: ["librlib"], |
| crate_name: "foo", |
| } |
| rust_ffi_static { |
| name: "libbaz", |
| srcs: ["foo.rs"], |
| rlibs: ["librlib"], |
| crate_name: "baz", |
| } |
| rust_library_rlib { |
| name: "librlib", |
| srcs: ["foo.rs"], |
| crate_name: "rlib", |
| } |
| rust_binary { |
| name: "fizz", |
| rlibs: ["librlib"], |
| static_libs: ["libbaz"], |
| srcs: ["foo.rs"], |
| } |
| cc_binary { |
| name: "buzz", |
| static_libs: ["libbaz"], |
| srcs: ["foo.c"], |
| } |
| cc_library { |
| name: "libbar", |
| static_libs: ["libbaz"], |
| compile_multilib: "64", |
| srcs: ["foo.c"], |
| }`) |
| |
| fizzZipInputs := ctx.ModuleForTests("fizz", "android_arm64_armv8-a_cov").Rule("zip").Inputs.Strings() |
| libfooZipInputs := ctx.ModuleForTests("libfoo", "android_arm64_armv8-a_dylib_cov").Rule("zip").Inputs.Strings() |
| buzzZipInputs := ctx.ModuleForTests("buzz", "android_arm64_armv8-a_cov").Rule("zip").Inputs.Strings() |
| libbarZipInputs := ctx.ModuleForTests("libbar", "android_arm64_armv8-a_shared_cov").Rule("zip").Inputs.Strings() |
| |
| // Make sure the expected number of input files are included. |
| if len(fizzZipInputs) != 3 { |
| t.Fatalf("expected only 3 coverage inputs for rust 'fizz' binary, got %#v: %#v", len(fizzZipInputs), fizzZipInputs) |
| } |
| if len(libfooZipInputs) != 2 { |
| t.Fatalf("expected only 2 coverage inputs for rust 'libfoo' library, got %#v: %#v", len(libfooZipInputs), libfooZipInputs) |
| } |
| if len(buzzZipInputs) != 2 { |
| t.Fatalf("expected only 2 coverage inputs for cc 'buzz' binary, got %#v: %#v", len(buzzZipInputs), buzzZipInputs) |
| } |
| if len(libbarZipInputs) != 2 { |
| t.Fatalf("expected only 2 coverage inputs for cc 'libbar' library, got %#v: %#v", len(libbarZipInputs), libbarZipInputs) |
| } |
| |
| // Make sure the expected inputs are provided to the zip rule. |
| if !android.SuffixInList(fizzZipInputs, "android_arm64_armv8-a_rlib_cov/librlib.gcno") || |
| !android.SuffixInList(fizzZipInputs, "android_arm64_armv8-a_static_cov/libbaz.gcno") || |
| !android.SuffixInList(fizzZipInputs, "android_arm64_armv8-a_cov/fizz.gcno") { |
| t.Fatalf("missing expected coverage files for rust 'fizz' binary: %#v", fizzZipInputs) |
| } |
| if !android.SuffixInList(libfooZipInputs, "android_arm64_armv8-a_rlib_cov/librlib.gcno") || |
| !android.SuffixInList(libfooZipInputs, "android_arm64_armv8-a_dylib_cov/libfoo.dylib.gcno") { |
| t.Fatalf("missing expected coverage files for rust 'fizz' binary: %#v", libfooZipInputs) |
| } |
| if !android.SuffixInList(buzzZipInputs, "android_arm64_armv8-a_cov/obj/foo.gcno") || |
| !android.SuffixInList(buzzZipInputs, "android_arm64_armv8-a_static_cov/libbaz.gcno") { |
| t.Fatalf("missing expected coverage files for cc 'buzz' binary: %#v", buzzZipInputs) |
| } |
| if !android.SuffixInList(libbarZipInputs, "android_arm64_armv8-a_static_cov/obj/foo.gcno") || |
| !android.SuffixInList(libbarZipInputs, "android_arm64_armv8-a_static_cov/libbaz.gcno") { |
| t.Fatalf("missing expected coverage files for cc 'libbar' library: %#v", libbarZipInputs) |
| } |
| } |
| |
| func TestCoverageDeps(t *testing.T) { |
| ctx := testRustCov(t, ` |
| rust_binary { |
| name: "fizz", |
| srcs: ["foo.rs"], |
| }`) |
| |
| fizz := ctx.ModuleForTests("fizz", "android_arm64_armv8-a_cov").Rule("rustc") |
| if !strings.Contains(fizz.Args["linkFlags"], "libprofile-extras.a") { |
| t.Fatalf("missing expected coverage 'libprofile-extras' dependency in linkFlags: %#v", fizz.Args["linkFlags"]) |
| } |
| } |