Generate unsupported-item comments in `h_body`.

PiperOrigin-RevId: 477772319
diff --git a/cc_bindings_from_rs/lib.rs b/cc_bindings_from_rs/lib.rs
index 482c60f..309ee36 100644
--- a/cc_bindings_from_rs/lib.rs
+++ b/cc_bindings_from_rs/lib.rs
@@ -2,32 +2,15 @@
 // Exceptions. See /LICENSE for license information.
 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 
-use itertools::Itertools;
+use anyhow::{bail, Result};
 use proc_macro2::TokenStream;
 use quote::quote;
-use rustc_hir::{Item, ItemKind, Node};
 use rustc_interface::Queries;
+use rustc_middle::dep_graph::DepContext;
 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
-        }
-    })
-}
+use rustc_span::def_id::{LocalDefId, LOCAL_CRATE};
+use std::fmt::Display;
 
 pub struct GeneratedBindings {
     pub h_body: TokenStream,
@@ -45,22 +28,16 @@
         };
 
         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
-            }
+            let main_content = format_crate(tcx);
+            quote! { #top_comment #main_content }
         };
 
         Self { h_body }
     }
 }
 
-/// Helper (used by `main` and `test::run_compiler`) for invoking functions
-/// operating on `TyCtxt`.
+/// Helper (used by `bindings_driver` and `test::run_compiler`) for invoking
+/// functions operating on `TyCtxt`.
 pub fn enter_tcx<'tcx, F, T>(
     queries: &'tcx Queries<'tcx>,
     f: F,
@@ -73,12 +50,45 @@
     Ok(query_context.peek_mut().enter(f))
 }
 
+fn format_def(_tcx: TyCtxt, _def_id: LocalDefId) -> Result<TokenStream> {
+    bail!("Nothing works yet!")
+}
+
+fn format_unsupported_def(
+    tcx: TyCtxt,
+    local_def_id: LocalDefId,
+    err_msg: impl Display,
+) -> TokenStream {
+    let span = tcx.sess().source_map().span_to_embeddable_string(tcx.def_span(local_def_id));
+    let name = tcx.def_path_str(local_def_id.to_def_id());
+    let msg = format!("Error while generating bindings for `{name}` defined at {span}: {err_msg}");
+    quote! { __COMMENT__ #msg }
+}
+
+fn format_crate(tcx: TyCtxt) -> TokenStream {
+    let snippets =
+        tcx.exported_symbols(LOCAL_CRATE).iter().filter_map(move |(symbol, _)| match symbol {
+            ExportedSymbol::NonGeneric(def_id) => {
+                let def_id = def_id.expect_local(); // Exports are always from the local crate.
+                match format_def(tcx, def_id) {
+                    Ok(snippet) => Some(snippet),
+                    Err(err) => Some(format_unsupported_def(tcx, def_id, err)),
+                }
+            }
+            ExportedSymbol::Generic(def_id, ..) => {
+                let def_id = def_id.expect_local(); // Exports are always from the local crate.
+                Some(format_unsupported_def(tcx, def_id, "Generics are not supported yet."))
+            }
+            ExportedSymbol::DropGlue(_) | ExportedSymbol::NoDefId(_) => None,
+        });
+    quote! { #( #snippets )* }
+}
+
 #[cfg(test)]
 pub mod tests {
     use super::GeneratedBindings;
 
     use anyhow::Result;
-    use itertools::Itertools;
     use std::path::PathBuf;
 
     use token_stream_printer::tokens_to_string;
@@ -93,23 +103,9 @@
     }
 
     #[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");
+        run_compiler("syntax error here", |_tcx| ())
     }
 
     #[test]
@@ -130,17 +126,14 @@
                 "// Automatically @generated C++ bindings for the following Rust crate:\n\
                  // rust_out\n\
                  \n\
-                 // List of public functions:\n\
-                 // public_function\n"
+                 // Error while generating bindings for `public_function` \
+                         defined at <crubit_unittests.rs>:2:17: 2:41: \
+                         Nothing works yet!\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,