Internal change

PiperOrigin-RevId: 545755024
Change-Id: I769ac75199651d9e9e19c9a09e41028e7abfc66d
diff --git a/cc_bindings_from_rs/bindings.rs b/cc_bindings_from_rs/bindings.rs
index af804a7..4c971dc 100644
--- a/cc_bindings_from_rs/bindings.rs
+++ b/cc_bindings_from_rs/bindings.rs
@@ -965,12 +965,16 @@
 
     /// Instance method taking `self` by value (i.e. `self: Self`).
     MethodTakingSelfByValue,
+
+    /// Instance method taking `self` by reference (i.e. `&self` or `&mut
+    /// self`).
+    MethodTakingSelfByRef,
 }
 
 impl FunctionKind {
     fn has_self_param(&self) -> bool {
         match self {
-            FunctionKind::MethodTakingSelfByValue => true,
+            FunctionKind::MethodTakingSelfByValue | FunctionKind::MethodTakingSelfByRef => true,
             FunctionKind::Free | FunctionKind::StaticMethod => false,
         }
     }
@@ -1052,17 +1056,32 @@
                 if params[0].ty == self_ty {
                     FunctionKind::MethodTakingSelfByValue
                 } else {
-                    bail!("Unsupported `self` type");
+                    match params[0].ty.kind() {
+                        ty::TyKind::Ref(_, referent_ty, _) if *referent_ty == self_ty => {
+                            FunctionKind::MethodTakingSelfByRef
+                        }
+                        _ => bail!("Unsupported `self` type"),
+                    }
                 }
             }
             _ => FunctionKind::StaticMethod,
         },
         other => panic!("Unexpected HIR node kind: {other:?}"),
     };
-    let method_qualifiers = if method_kind == FunctionKind::MethodTakingSelfByValue {
-        quote! { && }
-    } else {
-        quote! {}
+    let method_qualifiers = match method_kind {
+        FunctionKind::Free | FunctionKind::StaticMethod => quote! {},
+        FunctionKind::MethodTakingSelfByValue => quote! { && },
+        FunctionKind::MethodTakingSelfByRef => match params[0].ty.kind() {
+            ty::TyKind::Ref(region, _, mutability) => {
+                let lifetime_annotation = format_region_as_cc_lifetime(region);
+                let mutability = match mutability {
+                    Mutability::Mut => quote! {},
+                    Mutability::Not => quote! { const },
+                };
+                quote! { #mutability #lifetime_annotation }
+            }
+            _ => panic!("Expecting TyKind::Ref for MethodKind...Self...Ref"),
+        },
     };
 
     let struct_name = match self_ty {
@@ -1136,7 +1155,11 @@
             .enumerate()
             .map(|(i, Param { cc_name, ty, .. })| {
                 if i == 0 && method_kind.has_self_param() {
-                    quote! { this }
+                    if method_kind == FunctionKind::MethodTakingSelfByValue {
+                        quote! { this }
+                    } else {
+                        quote! { *this }
+                    }
                 } else if is_c_abi_compatible_by_value(*ty) {
                     quote! { #cc_name }
                 } else {
@@ -4558,8 +4581,51 @@
         test_format_item_method_taking_self_by_value(test_src);
     }
 
+    fn test_format_item_method_taking_self_by_const_ref(test_src: &str) {
+        test_format_item(test_src, "SomeStruct", |result| {
+            let result = result.unwrap().unwrap();
+            let main_api = &result.main_api;
+            assert_cc_matches!(
+                main_api.tokens,
+                quote! {
+                    ...
+                    struct ... SomeStruct final {
+                        ...
+                        inline
+                        float get_f32() const [[clang::annotate_type("lifetime", "__anon1")]];
+                        ...
+                    };
+                    ...
+                }
+            );
+            assert_cc_matches!(
+                result.cc_details.tokens,
+                quote! {
+                    namespace __crubit_internal {
+                    extern "C" float ...(
+                        ::rust_out::SomeStruct const& [[clang::annotate_type("lifetime",
+                                                                             "__anon1")]]);
+                    }
+                    inline float SomeStruct::get_f32()
+                        const [[clang::annotate_type("lifetime", "__anon1")]] {
+                      return __crubit_internal::...(*this);
+                    }
+                },
+            );
+            assert_rs_matches!(
+                result.rs_details,
+                quote! {
+                    #[no_mangle]
+                    extern "C" fn ...<'__anon1>(__self: &'__anon1 ::rust_out::SomeStruct) -> f32 {
+                        ::rust_out::SomeStruct::get_f32(__self)
+                    }
+                },
+            );
+        });
+    }
+
     #[test]
-    fn test_format_item_method_taking_self_by_const_ref() {
+    fn test_format_item_method_taking_self_by_const_ref_implicit_type() {
         let test_src = r#"
                 pub struct SomeStruct(f32);
 
@@ -4569,12 +4635,117 @@
                     }
                 }
             "#;
+        test_format_item_method_taking_self_by_const_ref(test_src);
+    }
+
+    #[test]
+    fn test_format_item_method_taking_self_by_const_ref_explicit_type() {
+        let test_src = r#"
+                pub struct SomeStruct(f32);
+
+                impl SomeStruct {
+                    pub fn get_f32(self: &SomeStruct) -> f32 {
+                        self.0
+                    }
+                }
+            "#;
+        test_format_item_method_taking_self_by_const_ref(test_src);
+    }
+
+    fn test_format_item_method_taking_self_by_mutable_ref(test_src: &str) {
+        test_format_item(test_src, "SomeStruct", |result| {
+            let result = result.unwrap().unwrap();
+            let main_api = &result.main_api;
+            assert_cc_matches!(
+                main_api.tokens,
+                quote! {
+                    ...
+                    struct ... SomeStruct final {
+                        ...
+                        inline void set_f32(float new_value)
+                            [[clang::annotate_type("lifetime", "__anon1")]];
+                        ...
+                    };
+                    ...
+                }
+            );
+            assert_cc_matches!(
+                result.cc_details.tokens,
+                quote! {
+                    namespace __crubit_internal {
+                    extern "C" void ...(
+                        ::rust_out::SomeStruct& [[clang::annotate_type("lifetime", "__anon1")]],
+                        float);
+                    }
+                    inline void SomeStruct::set_f32(float new_value)
+                            [[clang::annotate_type("lifetime", "__anon1")]] {
+                      return __crubit_internal::...(*this, new_value);
+                    }
+                },
+            );
+            assert_rs_matches!(
+                result.rs_details,
+                quote! {
+                    #[no_mangle]
+                    extern "C" fn ...<'__anon1>(
+                        __self: &'__anon1 mut ::rust_out::SomeStruct,
+                        new_value: f32
+                    ) -> () {
+                        ::rust_out::SomeStruct::set_f32(__self, new_value)
+                    }
+                },
+            );
+        });
+    }
+
+    #[test]
+    fn test_format_item_method_taking_self_by_mutable_ref_implicit_type() {
+        let test_src = r#"
+                pub struct SomeStruct(f32);
+
+                impl SomeStruct {
+                    pub fn set_f32(&mut self, new_value: f32) {
+                        self.0 = new_value;
+                    }
+                }
+            "#;
+        test_format_item_method_taking_self_by_mutable_ref(test_src);
+    }
+
+    #[test]
+    fn test_format_item_method_taking_self_by_mutable_ref_explicit_type() {
+        let test_src = r#"
+                pub struct SomeStruct(f32);
+
+                impl SomeStruct {
+                    pub fn set_f32(self: &mut SomeStruct, new_value: f32) {
+                        self.0 = new_value;
+                    }
+                }
+            "#;
+        test_format_item_method_taking_self_by_mutable_ref(test_src);
+    }
+
+    #[test]
+    fn test_format_item_method_taking_self_by_arc() {
+        let test_src = r#"
+                use std::sync::Arc;
+
+                pub struct SomeStruct(f32);
+
+                impl SomeStruct {
+                    pub fn get_f32(self: Arc<Self>) -> f32 {
+                        self.0
+                    }
+                }
+            "#;
         test_format_item(test_src, "SomeStruct", |result| {
             let result = result.unwrap().unwrap();
             let main_api = &result.main_api;
             let unsupported_msg = "Error generating bindings for `SomeStruct::get_f32` \
-                                   defined at <crubit_unittests.rs>;l=5: \
-                                   Unsupported `self` type";
+                                   defined at <crubit_unittests.rs>;l=7: \
+                                   Error handling parameter #0: \
+                                   Generic types are not supported yet (b/259749095)";
             assert_cc_matches!(
                 main_api.tokens,
                 quote! {
@@ -4593,40 +4764,6 @@
     }
 
     #[test]
-    fn test_format_item_method_taking_self_by_mutable_ref() {
-        let test_src = r#"
-                pub struct SomeStruct(f32);
-
-                impl SomeStruct {
-                    pub fn set_f32(&mut self, new_value: f32) {
-                        self.0 = new_value;
-                    }
-                }
-            "#;
-        test_format_item(test_src, "SomeStruct", |result| {
-            let result = result.unwrap().unwrap();
-            let main_api = &result.main_api;
-            let unsupported_msg = "Error generating bindings for `SomeStruct::set_f32` \
-                                   defined at <crubit_unittests.rs>;l=5: \
-                                   Unsupported `self` type";
-            assert_cc_matches!(
-                main_api.tokens,
-                quote! {
-                    ...
-                    struct ... SomeStruct final {
-                        ...
-                        __COMMENT__ #unsupported_msg
-                        ...
-                    };
-                    ...
-                }
-            );
-            assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::set_f32 },);
-            assert_rs_not_matches!(result.rs_details, quote! { set_f32 },);
-        });
-    }
-
-    #[test]
     fn test_format_item_method_taking_self_by_pinned_mut_ref() {
         let test_src = r#"
                 use core::pin::Pin;