1use binder::unstable_api::new_spibinder;
18use binder::{FromIBinder, SpIBinder, StatusCode, Strong};
19use foreign_types::{foreign_type, ForeignType, ForeignTypeRef};
20use std::os::fd::RawFd;
21use std::os::raw::{c_int, c_void};
22
23pub use binder_rpc_unstable_bindgen::ARpcSession_FileDescriptorTransportMode as FileDescriptorTransportMode;
24
25foreign_type! {
26 type CType = binder_rpc_unstable_bindgen::ARpcSession;
27 fn drop = binder_rpc_unstable_bindgen::ARpcSession_free;
28
29 #[derive(Debug)]
31 pub struct RpcSession;
32 pub struct RpcSessionRef;
34}
35
36unsafe impl Send for RpcSession {}
38unsafe impl Sync for RpcSession {}
40
41impl RpcSession {
42 pub fn new() -> RpcSession {
44 unsafe { RpcSession::from_ptr(binder_rpc_unstable_bindgen::ARpcSession_new()) }
46 }
47}
48
49impl Default for RpcSession {
50 fn default() -> Self {
51 Self::new()
52 }
53}
54
55impl RpcSessionRef {
56 pub fn set_file_descriptor_transport_mode(&self, mode: FileDescriptorTransportMode) {
58 unsafe {
60 binder_rpc_unstable_bindgen::ARpcSession_setFileDescriptorTransportMode(
61 self.as_ptr(),
62 mode,
63 )
64 };
65 }
66
67 pub fn set_max_incoming_threads(&self, threads: usize) {
69 unsafe {
71 binder_rpc_unstable_bindgen::ARpcSession_setMaxIncomingThreads(self.as_ptr(), threads)
72 };
73 }
74
75 pub fn set_max_outgoing_connections(&self, connections: usize) {
77 unsafe {
79 binder_rpc_unstable_bindgen::ARpcSession_setMaxOutgoingConnections(
80 self.as_ptr(),
81 connections,
82 )
83 };
84 }
85
86 #[cfg(not(target_os = "trusty"))]
88 pub fn setup_vsock_client<T: FromIBinder + ?Sized>(
89 &self,
90 cid: u32,
91 port: u32,
92 ) -> Result<Strong<T>, StatusCode> {
93 let service = unsafe {
96 new_spibinder(binder_rpc_unstable_bindgen::ARpcSession_setupVsockClient(
97 self.as_ptr(),
98 cid,
99 port,
100 ))
101 };
102 Self::get_interface(service)
103 }
104
105 #[cfg(not(target_os = "trusty"))]
108 pub fn setup_unix_domain_client<T: FromIBinder + ?Sized>(
109 &self,
110 socket_name: &str,
111 ) -> Result<Strong<T>, StatusCode> {
112 let socket_name = match std::ffi::CString::new(socket_name) {
113 Ok(s) => s,
114 Err(e) => {
115 log::error!("Cannot convert {} to CString. Error: {:?}", socket_name, e);
116 return Err(StatusCode::NAME_NOT_FOUND);
117 }
118 };
119
120 let service = unsafe {
123 new_spibinder(binder_rpc_unstable_bindgen::ARpcSession_setupUnixDomainClient(
124 self.as_ptr(),
125 socket_name.as_ptr(),
126 ))
127 };
128 Self::get_interface(service)
129 }
130
131 #[cfg(not(target_os = "trusty"))]
134 pub fn setup_unix_domain_bootstrap_client<T: FromIBinder + ?Sized>(
135 &self,
136 bootstrap_fd: std::os::fd::BorrowedFd,
137 ) -> Result<Strong<T>, StatusCode> {
138 use std::os::fd::AsRawFd;
139 let service = unsafe {
143 new_spibinder(binder_rpc_unstable_bindgen::ARpcSession_setupUnixDomainBootstrapClient(
144 self.as_ptr(),
145 bootstrap_fd.as_raw_fd(),
146 ))
147 };
148 Self::get_interface(service)
149 }
150
151 #[cfg(not(target_os = "trusty"))]
153 pub fn setup_inet_client<T: FromIBinder + ?Sized>(
154 &self,
155 address: &str,
156 port: u32,
157 ) -> Result<Strong<T>, StatusCode> {
158 let address = match std::ffi::CString::new(address) {
159 Ok(s) => s,
160 Err(e) => {
161 log::error!("Cannot convert {} to CString. Error: {:?}", address, e);
162 return Err(StatusCode::BAD_VALUE);
163 }
164 };
165
166 let service = unsafe {
169 new_spibinder(binder_rpc_unstable_bindgen::ARpcSession_setupInet(
170 self.as_ptr(),
171 address.as_ptr(),
172 port,
173 ))
174 };
175 Self::get_interface(service)
176 }
177
178 #[cfg(target_os = "trusty")]
179 pub fn setup_trusty_client<T: FromIBinder + ?Sized>(
180 &self,
181 port: &std::ffi::CStr,
182 ) -> Result<Strong<T>, StatusCode> {
183 self.setup_preconnected_client(|| {
184 let h = tipc::Handle::connect(port)
185 .expect("Failed to connect to service port {SERVICE_PORT}");
186
187 let fd = h.as_raw_fd();
189 core::mem::forget(h);
190 Some(fd)
191 })
192 }
193
194 pub fn setup_preconnected_client<T: FromIBinder + ?Sized>(
197 &self,
198 mut request_fd: impl FnMut() -> Option<RawFd>,
199 ) -> Result<Strong<T>, StatusCode> {
200 let mut request_fd_ref: RequestFd = &mut request_fd;
202 let param = &mut request_fd_ref as *mut RequestFd as *mut c_void;
203
204 let service = unsafe {
208 new_spibinder(binder_rpc_unstable_bindgen::ARpcSession_setupPreconnectedClient(
209 self.as_ptr(),
210 Some(request_fd_wrapper),
211 param,
212 ))
213 };
214 Self::get_interface(service)
215 }
216
217 fn get_interface<T: FromIBinder + ?Sized>(
218 service: Option<SpIBinder>,
219 ) -> Result<Strong<T>, StatusCode> {
220 if let Some(service) = service {
221 FromIBinder::try_from(service)
222 } else {
223 Err(StatusCode::NAME_NOT_FOUND)
224 }
225 }
226}
227
228type RequestFd<'a> = &'a mut dyn FnMut() -> Option<RawFd>;
229
230unsafe extern "C" fn request_fd_wrapper(param: *mut c_void) -> c_int {
231 let request_fd_ptr = param as *mut RequestFd;
232 let request_fd = unsafe { request_fd_ptr.as_mut().unwrap() };
236 request_fd().unwrap_or(-1)
237}