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