blob: 16c3c3816e3ddeb0675bf36df95607b32e05db55 [file]
/*
* Copyright (C) 2020 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
//! Safe Rust interface to Android `libbinder`.
//!
//! This crate is primarily designed as an target for a Rust AIDL compiler
//! backend, and should generally not be used directly by users. It is built on
//! top of the binder NDK library to be usable by APEX modules, and therefore
//! only exposes functionality available in the NDK interface.
//!
//! # Example
//!
//! The following example illustrates how the AIDL backend will use this crate.
//!
//! ```
//! use binder::{
//! declare_binder_interface, Binder, IBinder, Interface, Remotable, Parcel, SpIBinder,
//! StatusCode, TransactionCode,
//! };
//!
//! // Generated by AIDL compiler
//! pub trait ITest: Interface {
//! fn test(&self) -> binder::Result<String>;
//! }
//!
//! // Creates a new local (native) service object, BnTest, and a remote proxy
//! // object, BpTest, that are the typed interfaces for their respective ends
//! // of the binder transaction. Generated by AIDL compiler.
//! declare_binder_interface! {
//! ITest["android.os.ITest"] {
//! native: BnTest(on_transact),
//! proxy: BpTest,
//! }
//! }
//!
//! // Generated by AIDL compiler
//! fn on_transact(
//! service: &dyn ITest,
//! code: TransactionCode,
//! _data: &BorrowedParcel,
//! reply: &mut BorrowedParcel,
//! ) -> binder::Result<()> {
//! match code {
//! SpIBinder::FIRST_CALL_TRANSACTION => {
//! reply.write(&service.test()?)?;
//! Ok(())
//! }
//! _ => Err(StatusCode::UNKNOWN_TRANSACTION),
//! }
//! }
//!
//! // Generated by AIDL compiler
//! impl ITest for Binder<BnTest> {
//! fn test(&self) -> binder::Result<String> {
//! self.0.test()
//! }
//! }
//!
//! // Generated by AIDL compiler
//! impl ITest for BpTest {
//! fn test(&self) -> binder::Result<String> {
//! let reply = self
//! .as_binder()
//! .transact(SpIBinder::FIRST_CALL_TRANSACTION, 0, |_| Ok(()))?;
//! reply.read()
//! }
//! }
//!
//! // User implemented:
//!
//! // Local implementation of the ITest remotable interface.
//! struct TestService;
//!
//! impl Interface for TestService {}
//!
//! impl ITest for TestService {
//! fn test(&self) -> binder::Result<String> {
//! Ok("testing service".to_string())
//! }
//! }
//! ```
#![cfg_attr(not(feature = "std"), no_std)]
#[macro_use]
extern crate alloc;
#[cfg(all(feature = "std", not(any(android_ndk, trusty))))]
mod accessor;
#[macro_use]
mod binder;
mod binder_async;
mod error;
mod native;
mod parcel;
#[cfg(not(trusty))]
mod persistable_bundle;
mod proxy;
#[cfg(all(feature = "kernel_ipc", not(android_ndk)))]
mod service;
#[cfg(all(feature = "kernel_ipc", not(android_ndk)))]
mod state;
#[macro_use]
mod sync_utils;
mod write_to;
use binder_ndk_sys as sys;
pub use crate::binder_async::{BinderAsyncPool, BoxFuture};
#[cfg(all(feature = "std", not(any(android_ndk, trusty))))]
pub use accessor::{delegate_accessor, Accessor, AccessorProvider, ConnectionInfo};
pub use binder::{BinderFeatures, FromIBinder, IBinder, Interface, Strong, Weak};
pub use error::{ExceptionCode, IntoBinderResult, Status, StatusCode};
pub use parcel::{ParcelFileDescriptor, Parcelable, ParcelableHolder};
#[cfg(not(trusty))]
pub use persistable_bundle::{PersistableBundle, ValueType};
pub use proxy::{DeathRecipient, SpIBinder, WpIBinder};
#[cfg(all(feature = "kernel_ipc", not(android_ndk)))]
pub use service::{
add_service, check_interface, check_service, force_lazy_services_persist,
get_declared_instances, is_declared, is_handling_transaction, register_lazy_service,
wait_for_interface, wait_for_service, LazyServiceGuard,
};
// TODO(b/402766978) Once LLDNK symbols are supported in rust, this can be along with the rest
// of the service symbols in vendor variants.
#[cfg(all(feature = "kernel_ipc", not(any(android_ndk, android_vendor, android_vndk))))]
pub use service::{
check_service_access, CHECK_ACCESS_PERMISSION_ADD, CHECK_ACCESS_PERMISSION_FIND,
CHECK_ACCESS_PERMISSION_LIST,
};
#[cfg(all(feature = "kernel_ipc", not(android_ndk)))]
#[allow(deprecated)]
pub use service::{get_interface, get_service};
#[cfg(all(feature = "kernel_ipc", not(android_ndk)))]
pub use state::{ProcessState, ThreadState};
pub use write_to::WriteTo;
/// Binder result containing a [`Status`] on error.
pub type Result<T> = core::result::Result<T, Status>;
/// Advanced Binder APIs needed internally by AIDL or when manually using Binder
/// without AIDL.
pub mod binder_impl {
pub use crate::binder::{
FunctionNames, IBinderInternal, InterfaceClass, LocalStabilityType, Remotable, Stability,
StabilityType, ToAsyncInterface, ToSyncInterface, TransactionCode, TransactionFlags,
VintfStabilityType, FIRST_CALL_TRANSACTION, FLAG_ONEWAY, LAST_CALL_TRANSACTION,
};
#[cfg(not(android_ndk))]
pub use crate::binder::{FLAG_CLEAR_BUF, FLAG_PRIVATE_LOCAL};
pub use crate::binder_async::BinderAsyncRuntime;
pub use crate::error::status_t;
pub use crate::native::Binder;
pub use crate::parcel::{
BorrowedParcel, Deserialize, DeserializeArray, DeserializeOption, Parcel,
ParcelableMetadata, Serialize, SerializeArray, SerializeOption, UnstructuredParcelable,
NON_NULL_PARCELABLE_FLAG, NULL_PARCELABLE_FLAG,
};
pub use crate::proxy::{AssociateClass, Proxy};
pub use crate::panic_if_poisoned;
#[cfg(single_threaded)]
pub use spin::Mutex;
#[cfg(all(feature = "std", not(single_threaded)))]
pub use std::sync::Mutex;
#[cfg(single_threaded)]
pub use spin::Once;
#[cfg(all(feature = "std", not(single_threaded)))]
pub use std::sync::Once;
}
/// Unstable, in-development API that only allowlisted clients are allowed to use.
#[doc(hidden)]
pub mod unstable_api {
pub use crate::binder::AsNative;
pub use crate::error::status_result;
pub use crate::proxy::unstable_api::new_spibinder;
pub use crate::sys::AIBinder;
pub use crate::sys::AParcel;
}