| // Part of the Crubit project, under the Apache License v2.0 with LLVM |
| // Exceptions. See /LICENSE for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| |
| use itertools::Itertools; |
| use proc_macro2::TokenStream; |
| use quote::quote; |
| use rustc_hir::{Item, ItemKind, Node}; |
| use rustc_interface::Queries; |
| use rustc_middle::middle::exported_symbols::ExportedSymbol; |
| use rustc_middle::ty::TyCtxt; |
| use rustc_span::def_id::LOCAL_CRATE; |
| |
| // TODO(lukasza): Replace `get_names_of_exported_fns` with something that can |
| // generate C++ bindings. |
| fn get_names_of_exported_fns(tcx: TyCtxt) -> impl Iterator<Item = String> + '_ { |
| tcx.exported_symbols(LOCAL_CRATE).iter().filter_map(move |(symbol, _)| match symbol { |
| ExportedSymbol::NonGeneric(def_id) => { |
| match tcx.hir().get_by_def_id(def_id.expect_local()) { |
| Node::Item(Item { kind: ItemKind::Fn { .. }, .. }) => { |
| Some(tcx.def_path_str(*def_id)) |
| } |
| _ => None, |
| } |
| } |
| ExportedSymbol::Generic(..) | ExportedSymbol::DropGlue(_) | ExportedSymbol::NoDefId(_) => { |
| None |
| } |
| }) |
| } |
| |
| pub struct GeneratedBindings { |
| pub h_body: TokenStream, |
| } |
| |
| impl GeneratedBindings { |
| pub fn generate(tcx: TyCtxt) -> Self { |
| let top_comment = { |
| let crate_name = tcx.crate_name(LOCAL_CRATE); |
| let txt = format!( |
| "Automatically @generated C++ bindings for the following Rust crate:\n\ |
| {crate_name}" |
| ); |
| quote! { __COMMENT__ #txt __NEWLINE__ } |
| }; |
| |
| let h_body = { |
| let comment_with_names_of_public_functions = format!( |
| "List of public functions:\n{}", |
| get_names_of_exported_fns(tcx).collect_vec().join(",\n") |
| ); |
| quote! { |
| #top_comment |
| __COMMENT__ #comment_with_names_of_public_functions |
| } |
| }; |
| |
| Self { h_body } |
| } |
| } |
| |
| /// Helper (used by `main` and `test::run_compiler`) for invoking functions |
| /// operating on `TyCtxt`. |
| pub fn enter_tcx<'tcx, F, T>( |
| queries: &'tcx Queries<'tcx>, |
| f: F, |
| ) -> rustc_interface::interface::Result<T> |
| where |
| F: FnOnce(TyCtxt<'tcx>) -> T + Send, |
| T: Send, |
| { |
| let query_context = queries.global_ctxt()?; |
| Ok(query_context.peek_mut().enter(f)) |
| } |
| |
| #[cfg(test)] |
| mod tests { |
| use super::GeneratedBindings; |
| |
| use anyhow::Result; |
| use itertools::Itertools; |
| |
| use token_stream_printer::tokens_to_string; |
| |
| #[test] |
| fn test_get_names_of_exported_fns_public_vs_private() { |
| let test_src = r#" |
| pub fn public_function() { |
| private_function() |
| } |
| |
| fn private_function() {} |
| "#; |
| let exported_functions = get_names_of_exported_fns(test_src); |
| assert_eq!(1, exported_functions.len()); |
| assert_eq!("public_function", exported_functions[0]); |
| } |
| |
| #[test] |
| #[should_panic] |
| fn test_panic_when_syntax_errors_in_test_inputs() { |
| get_names_of_exported_fns("syntax error here"); |
| } |
| |
| #[test] |
| fn test_generated_bindings_happy_path() -> Result<()> { |
| let test_src = r#" |
| pub fn public_function() { |
| private_function() |
| } |
| |
| fn private_function() {} |
| "#; |
| test_generated_bindings(test_src, |bindings| { |
| // TODO(lukasza): Use `assert_cc_matches!` from |
| // `rs_bindings_from_cc/token_stream_matchers.rs` here. |
| let h_body = tokens_to_string(bindings.h_body)?; |
| assert_eq!( |
| h_body, |
| "// Automatically @generated C++ bindings for the following Rust crate:\n\ |
| // rust_out\n\ |
| \n\ |
| // List of public functions:\n\ |
| // public_function\n" |
| ); |
| Ok(()) |
| }) |
| } |
| |
| fn get_names_of_exported_fns(source: &str) -> Vec<String> { |
| run_compiler(source, |tcx| super::get_names_of_exported_fns(tcx).collect_vec()) |
| } |
| |
| fn test_generated_bindings<F, T>(source: &str, f: F) -> T |
| where |
| F: FnOnce(GeneratedBindings) -> T + Send, |
| T: Send, |
| { |
| run_compiler(source, |tcx| f(GeneratedBindings::generate(tcx))) |
| } |
| |
| /// Compiles Rust `source` then calls `f` on the `TyCtxt` representation |
| /// of the compiled `source`. |
| fn run_compiler<F, T>(source: impl Into<String>, f: F) -> T |
| where |
| F: for<'tcx> FnOnce(rustc_middle::ty::TyCtxt<'tcx>) -> T + Send, |
| T: Send, |
| { |
| use rustc_session::config::{CrateType, Input, Options, OutputType, OutputTypes}; |
| |
| const TEST_FILENAME: &str = "crubit_unittests.rs"; |
| |
| // Setting `output_types` that will trigger code gen - otherwise some parts of |
| // the analysis will be missing (e.g. `tcx.exported_symbols()`). |
| // The choice of `Bitcode` is somewhat arbitrary (e.g. `Assembly`, |
| // `Mir`, etc. would also trigger code gen). |
| let output_types = OutputTypes::new(&[(OutputType::Bitcode, None /* PathBuf */)]); |
| |
| let runfiles = runfiles::Runfiles::create().unwrap(); |
| let sysroot_path = |
| runfiles.rlocation(if std::env::var("LEGACY_TOOLCHAIN_RUST_TEST").is_ok() { |
| "google3/third_party/unsupported_toolchains/rust/toolchains/nightly" |
| } else { |
| "google3/nowhere/llvm/rust" |
| }); |
| let opts = Options { |
| crate_types: vec![CrateType::Rlib], // Test inputs simulate library crates. |
| maybe_sysroot: Some(sysroot_path), |
| output_types, |
| ..Default::default() |
| }; |
| |
| let config = rustc_interface::interface::Config { |
| opts, |
| crate_cfg: Default::default(), |
| crate_check_cfg: Default::default(), |
| input: Input::Str { |
| name: rustc_span::FileName::Custom(TEST_FILENAME.to_string()), |
| input: source.into(), |
| }, |
| input_path: None, |
| output_file: None, |
| output_dir: None, |
| file_loader: None, |
| diagnostic_output: rustc_session::DiagnosticOutput::Default, |
| lint_caps: Default::default(), |
| parse_sess_created: None, |
| register_lints: None, |
| override_queries: None, |
| make_codegen_backend: None, |
| registry: rustc_errors::registry::Registry::new(rustc_error_codes::DIAGNOSTICS), |
| }; |
| |
| rustc_interface::interface::run_compiler(config, |compiler| { |
| compiler.enter(|queries| super::enter_tcx(queries, f)) |
| }) |
| .expect("Test inputs shouldn't cause compilation errors.") |
| } |
| } |