// 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" ) func TestRustBindgen(t *testing.T) { ctx := testRust(t, ` rust_bindgen { name: "libbindgen", defaults: ["cc_defaults_flags"], wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", bindgen_flags: ["--bindgen-flag.*"], cflags: ["--clang-flag()"], shared_libs: ["libfoo_shared"], } rust_bindgen { name: "libbindgen_staticlib", wrapper_src: "src/any.h", crate_name: "bindgen_staticlib", stem: "libbindgen_staticlib", source_stem: "bindings", static_libs: ["libfoo_static"], } rust_bindgen { name: "libbindgen_headerlib", wrapper_src: "src/any.h", crate_name: "bindgen_headerlib", stem: "libbindgen_headerlib", source_stem: "bindings", header_libs: ["libfoo_header"], } cc_library_shared { name: "libfoo_shared", export_include_dirs: ["shared_include"], } cc_library_static { name: "libfoo_static", export_include_dirs: ["static_include"], } cc_library_headers { name: "libfoo_header", export_include_dirs: ["header_include"], } cc_defaults { name: "cc_defaults_flags", cflags: ["--default-flag"], } `) libbindgen := ctx.ModuleForTests("libbindgen", "android_arm64_armv8-a_source").Output("bindings.rs") libbindgenStatic := ctx.ModuleForTests("libbindgen_staticlib", "android_arm64_armv8-a_source").Output("bindings.rs") libbindgenHeader := ctx.ModuleForTests("libbindgen_headerlib", "android_arm64_armv8-a_source").Output("bindings.rs") libbindgenHeaderModule := ctx.ModuleForTests("libbindgen_headerlib", "android_arm64_armv8-a_source").Module().(*Module) // Ensure that the flags are present and escaped if !strings.Contains(libbindgen.Args["flags"], "'--bindgen-flag.*'") { t.Errorf("missing bindgen flags in rust_bindgen rule: flags %#v", libbindgen.Args["flags"]) } if !strings.Contains(libbindgen.Args["cflags"], "'--clang-flag()'") { t.Errorf("missing clang cflags in rust_bindgen rule: cflags %#v", libbindgen.Args["cflags"]) } if !strings.Contains(libbindgen.Args["cflags"], "-Ishared_include") { t.Errorf("missing shared_libs exported includes in rust_bindgen rule: cflags %#v", libbindgen.Args["cflags"]) } if !strings.Contains(libbindgenStatic.Args["cflags"], "-Istatic_include") { t.Errorf("missing static_libs exported includes in rust_bindgen rule: cflags %#v", libbindgenStatic.Args["cflags"]) } if !strings.Contains(libbindgenHeader.Args["cflags"], "-Iheader_include") { t.Errorf("missing header_libs exported includes in rust_bindgen rule: cflags %#v", libbindgenHeader.Args["cflags"]) } if android.InList("libfoo_static", libbindgenHeaderModule.Properties.AndroidMkHeaderLibs) { t.Errorf("Static library dependency should not be in HeaderLibs list") } if !strings.Contains(libbindgen.Args["cflags"], "--default-flag") { t.Errorf("rust_bindgen missing cflags defined in cc_defaults: cflags %#v", libbindgen.Args["cflags"]) } } func TestRustBindgenCustomBindgen(t *testing.T) { ctx := testRust(t, ` rust_bindgen { name: "libbindgen", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", custom_bindgen: "my_bindgen" } rust_binary_host { name: "my_bindgen", srcs: ["foo.rs"], } `) libbindgen := ctx.ModuleForTests("libbindgen", "android_arm64_armv8-a_source").Output("bindings.rs") // The rule description should contain the custom binary name rather than bindgen, so checking the description // should be sufficient. if !strings.Contains(libbindgen.Description, "my_bindgen") { t.Errorf("Custom bindgen binary %s not used for libbindgen: rule description %#v", "my_bindgen", libbindgen.Description) } } func TestRustBindgenStdVersions(t *testing.T) { testRustError(t, "c_std and cpp_std cannot both be defined at the same time.", ` rust_bindgen { name: "libbindgen", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", c_std: "somevalue", cpp_std: "somevalue", } `) ctx := testRust(t, ` rust_bindgen { name: "libbindgen_cstd", wrapper_src: "src/any.hpp", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", c_std: "foo" } rust_bindgen { name: "libbindgen_cppstd", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", cpp_std: "foo" } `) libbindgen_cstd := ctx.ModuleForTests("libbindgen_cstd", "android_arm64_armv8-a_source").Output("bindings.rs") libbindgen_cppstd := ctx.ModuleForTests("libbindgen_cppstd", "android_arm64_armv8-a_source").Output("bindings.rs") if !strings.Contains(libbindgen_cstd.Args["cflags"], "-std=foo") { t.Errorf("c_std value not passed in to rust_bindgen as a clang flag") } if !strings.Contains(libbindgen_cppstd.Args["cflags"], "-std=foo") { t.Errorf("cpp_std value not passed in to rust_bindgen as a clang flag") } // Make sure specifying cpp_std emits the '-x c++' flag if !strings.Contains(libbindgen_cppstd.Args["cflags"], "-x c++") { t.Errorf("Setting cpp_std should cause the '-x c++' flag to be emitted") } // Make sure specifying c_std omits the '-x c++' flag if strings.Contains(libbindgen_cstd.Args["cflags"], "-x c++") { t.Errorf("Setting c_std should not cause the '-x c++' flag to be emitted") } } func TestBindgenDisallowedFlags(t *testing.T) { // Make sure passing '-x c++' to cflags generates an error testRustError(t, "cflags: -x c\\+\\+ should not be specified in cflags.*", ` rust_bindgen { name: "libbad_flag", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", cflags: ["-x c++"] } `) // Make sure passing '-std=' to cflags generates an error testRustError(t, "cflags: -std should not be specified in cflags.*", ` rust_bindgen { name: "libbad_flag", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", cflags: ["-std=foo"] } `) } func TestBindgenFlagFile(t *testing.T) { ctx := testRust(t, ` rust_bindgen { name: "libbindgen", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", bindgen_flag_files: [ "flag_file.txt", ], } `) libbindgen := ctx.ModuleForTests("libbindgen", "android_arm64_armv8-a_source").Output("bindings.rs") if !strings.Contains(libbindgen.Args["flagfiles"], "/dev/null") { t.Errorf("missing /dev/null in rust_bindgen rule: flags %#v", libbindgen.Args["flagfiles"]) } if !strings.Contains(libbindgen.Args["flagfiles"], "flag_file.txt") { t.Errorf("missing bindgen flags file in rust_bindgen rule: flags %#v", libbindgen.Args["flagfiles"]) } // TODO: The best we can do right now is check $flagfiles. Once bindgen.go switches to RuleBuilder, // we may be able to check libbinder.RuleParams.Command to see if it contains $(cat /dev/null flag_file.txt) } func TestBindgenHandleStaticInlining(t *testing.T) { ctx := testRust(t, ` rust_bindgen { name: "libbindgen", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", handle_static_inline: true, static_inline_library: "libbindgen_staticfns" } cc_library_static { name: "libbindgen_staticfns", srcs: [":libbindgen"], include_dirs: ["src/"], } `) libbindgen := ctx.ModuleForTests("libbindgen", "android_arm64_armv8-a_source").Output("bindings.rs") // Make sure the flag to support `static inline` functions is present if !strings.Contains(libbindgen.Args["flags"], "--wrap-static-fns") { t.Errorf("missing flag to handle static inlining in rust_bindgen rule: flags %#v", libbindgen.Args["flags"]) } if !strings.Contains(libbindgen.Args["flags"], "--wrap-static-fns-path") { t.Errorf("missing flag to define path for static inlining C source from bindgen (--wrap-static-fns-path): flags %#v", libbindgen.Args["flags"]) } } func TestBindgenStaticInlineProperties(t *testing.T) { // Make sure handle_static_inline without static_inline_library generates an error testRustError(t, "requires declaring static_inline_library to the corresponding cc_library module that includes the generated C source from bindgen", ` rust_bindgen { name: "libbindgen", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", handle_static_inline: true } `) testRustError(t, "requires declaring handle_static_inline", ` rust_bindgen { name: "libbindgen", wrapper_src: "src/any.h", crate_name: "bindgen", stem: "libbindgen", source_stem: "bindings", static_inline_library: "libbindgen_staticfns" } cc_library_static { name: "libbindgen_staticfns", srcs: [":libbindgen"], include_dirs: ["src/"], } `) }