Refactoring: deduplication: `format_param_types_for_cc`.

Quite similar code was used by both `format_fn` and `format_thunk_decl`
to format function parameters.  The duplication would get somewhat worse
after adding support for function pointers (which adds a 3rd place where
parameter types need to be formatted + requires formatting them with
a future/new TypeLocation).

Because of the above, this CL extracts the shared code into
`format_param_types_for_cc`.

PiperOrigin-RevId: 524116023
diff --git a/cc_bindings_from_rs/bindings.rs b/cc_bindings_from_rs/bindings.rs
index 1765130..465f673 100644
--- a/cc_bindings_from_rs/bindings.rs
+++ b/cc_bindings_from_rs/bindings.rs
@@ -505,6 +505,20 @@
         .context("Error formatting function return type")
 }
 
+fn format_param_types_for_cc<'tcx>(
+    input: &Input<'tcx>,
+    sig: &ty::FnSig<'tcx>,
+) -> Result<Vec<CcSnippet>> {
+    sig.inputs()
+        .iter()
+        .enumerate()
+        .map(|(i, &ty)| {
+            Ok(format_ty_for_cc(input, ty, TypeLocation::Other)
+                .with_context(|| format!("Error handling parameter #{i}"))?)
+        })
+        .collect()
+}
+
 /// Formats `ty` for Rust - to be used in `..._cc_api_impl.rs` (e.g. as a type
 /// of a parameter in a Rust thunk).  Because `..._cc_api_impl.rs` is a
 /// distinct, separate crate, the returned `TokenStream` uses crate-qualified
@@ -600,26 +614,29 @@
     let sig = get_fn_sig(tcx, fn_def_id)?;
     let main_api_ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut prereqs);
 
-    let mut thunk_params = sig
-        .inputs()
-        .iter()
-        .map(|&ty| -> Result<TokenStream> {
-            let cc_type = format_ty_for_cc(input, ty, TypeLocation::Other)?.into_tokens(&mut prereqs);
-            if is_c_abi_compatible_by_value(ty) {
-                Ok(quote! { #cc_type })
-            } else {
-                // Rust thunk will move a value via memcpy - we need to `ensure` that
-                // invoking the C++ destructor (on the moved-away value) is safe.
-                // TODO(b/259749095): Support generic structs (with non-empty ParamEnv).
-                ensure!(
-                    !ty.needs_drop(tcx, ty::ParamEnv::empty()),
-                    "Only trivially-movable and trivially-destructible types \
-                          may be passed by value over the FFI boundary"
-                );
-                Ok(quote! { #cc_type* })
-            }
-        })
-        .collect::<Result<Vec<_>>>()?;
+    let mut thunk_params = {
+        let cc_types = format_param_types_for_cc(input, &sig)?;
+        sig.inputs()
+            .iter()
+            .zip(cc_types.into_iter())
+            .map(|(&ty, cc_type)| -> Result<TokenStream> {
+                let cc_type = cc_type.into_tokens(&mut prereqs);
+                if is_c_abi_compatible_by_value(ty) {
+                    Ok(quote! { #cc_type })
+                } else {
+                    // Rust thunk will move a value via memcpy - we need to `ensure` that
+                    // invoking the C++ destructor (on the moved-away value) is safe.
+                    // TODO(b/259749095): Support generic structs (with non-empty ParamEnv).
+                    ensure!(
+                        !ty.needs_drop(tcx, ty::ParamEnv::empty()),
+                        "Only trivially-movable and trivially-destructible types \
+                              may be passed by value over the FFI boundary"
+                    );
+                    Ok(quote! { #cc_type* })
+                }
+            })
+            .collect::<Result<Vec<_>>>()?
+    };
 
     let thunk_ret_type: TokenStream;
     if is_c_abi_compatible_by_value(sig.output()) {
@@ -814,19 +831,18 @@
     }
     let params = {
         let names = tcx.fn_arg_names(def_id).iter();
-        let types = sig.inputs().iter();
+        let cc_types = format_param_types_for_cc(input, &sig)?;
         names
-            .zip(types)
             .enumerate()
-            .map(|(i, (name, &ty))| -> Result<Param> {
+            .zip(sig.inputs().iter())
+            .zip(cc_types.into_iter())
+            .map(|(((i, name), &ty), cc_type)| {
                 let cc_name = format_cc_ident(name.as_str())
                     .unwrap_or_else(|_err| format_cc_ident(&format!("__param_{i}")).unwrap());
-                let cc_type = format_ty_for_cc(input, ty, TypeLocation::Other)?.into_tokens(&mut main_api_prereqs);
-                Ok(Param { cc_name, cc_type, ty })
+                let cc_type = cc_type.into_tokens(&mut main_api_prereqs);
+                Param { cc_name, cc_type, ty }
             })
-            .enumerate()
-            .map(|(i, result)| result.with_context(|| format!("Error handling parameter #{i}")))
-            .collect::<Result<Vec<_>>>()?
+            .collect_vec()
     };
     let main_api_params = params
         .iter()