Skip to content

Commit c27e705

Browse files
Darksonnojeda
authored andcommittedJan 13, 2025
rust: kernel: add improved version of ForeignOwnable::borrow_mut
Previously, the `ForeignOwnable` trait had a method called `borrow_mut` that was intended to provide mutable access to the inner value. However, the method accidentally made it possible to change the address of the object being modified, which usually isn't what we want. (And when we want that, it can be done by calling `from_foreign` and `into_foreign`, like how the old `borrow_mut` was implemented.) In this patch, we introduce an alternate definition of `borrow_mut` that solves the previous problem. Conceptually, given a pointer type `P` that implements `ForeignOwnable`, the `borrow_mut` method gives you the same kind of access as an `&mut P` would, except that it does not let you change the pointer `P` itself. This is analogous to how the existing `borrow` method provides the same kind of access to the inner value as an `&P`. Note that for types like `Arc`, having an `&mut Arc<T>` only gives you immutable access to the inner `T`. This is because mutable references assume exclusive access, but there might be other handles to the same reference counted value, so the access isn't exclusive. The `Arc` type implements this by making `borrow_mut` return the same type as `borrow`. Signed-off-by: Alice Ryhl <aliceryhl@google.com> Reviewed-by: Boqun Feng <boqun.feng@gmail.com> Reviewed-by: Benno Lossin <benno.lossin@proton.me> Reviewed-by: Martin Rodriguez Reboredo <yakoyoku@gmail.com> Reviewed-by: Andreas Hindborg <a.hindborg@kernel.org> Signed-off-by: Tamir Duberstein <tamird@gmail.com> Acked-by: Danilo Krummrich <dakr@kernel.org> Link: https://lore.kernel.org/r/20241120-borrow-mut-v6-6-80dbadd00951@gmail.com [ Updated to `crate::ffi::`. Reworded title slightly. - Miguel ] Signed-off-by: Miguel Ojeda <ojeda@kernel.org>
1 parent c6b9753 commit c27e705

File tree

3 files changed

+86
-13
lines changed

3 files changed

+86
-13
lines changed
 

‎rust/kernel/alloc/kbox.rs

+21
Original file line numberDiff line numberDiff line change
@@ -354,6 +354,7 @@ where
354354
A: Allocator,
355355
{
356356
type Borrowed<'a> = &'a T;
357+
type BorrowedMut<'a> = &'a mut T;
357358

358359
fn into_foreign(self) -> *mut crate::ffi::c_void {
359360
Box::into_raw(self).cast()
@@ -370,13 +371,21 @@ where
370371
// immutable for the duration of 'a.
371372
unsafe { &*ptr.cast() }
372373
}
374+
375+
unsafe fn borrow_mut<'a>(ptr: *mut crate::ffi::c_void) -> &'a mut T {
376+
let ptr = ptr.cast();
377+
// SAFETY: The safety requirements of this method ensure that the pointer is valid and that
378+
// nothing else will access the value for the duration of 'a.
379+
unsafe { &mut *ptr }
380+
}
373381
}
374382

375383
impl<T: 'static, A> ForeignOwnable for Pin<Box<T, A>>
376384
where
377385
A: Allocator,
378386
{
379387
type Borrowed<'a> = Pin<&'a T>;
388+
type BorrowedMut<'a> = Pin<&'a mut T>;
380389

381390
fn into_foreign(self) -> *mut crate::ffi::c_void {
382391
// SAFETY: We are still treating the box as pinned.
@@ -399,6 +408,18 @@ where
399408
// SAFETY: This pointer originates from a `Pin<Box<T>>`.
400409
unsafe { Pin::new_unchecked(r) }
401410
}
411+
412+
unsafe fn borrow_mut<'a>(ptr: *mut crate::ffi::c_void) -> Pin<&'a mut T> {
413+
let ptr = ptr.cast();
414+
// SAFETY: The safety requirements for this function ensure that the object is still alive,
415+
// so it is safe to dereference the raw pointer.
416+
// The safety requirements of `from_foreign` also ensure that the object remains alive for
417+
// the lifetime of the returned value.
418+
let r = unsafe { &mut *ptr };
419+
420+
// SAFETY: This pointer originates from a `Pin<Box<T>>`.
421+
unsafe { Pin::new_unchecked(r) }
422+
}
402423
}
403424

404425
impl<T, A> Deref for Box<T, A>

‎rust/kernel/sync/arc.rs

+7
Original file line numberDiff line numberDiff line change
@@ -344,6 +344,7 @@ impl<T: ?Sized> Arc<T> {
344344

345345
impl<T: 'static> ForeignOwnable for Arc<T> {
346346
type Borrowed<'a> = ArcBorrow<'a, T>;
347+
type BorrowedMut<'a> = Self::Borrowed<'a>;
347348

348349
fn into_foreign(self) -> *mut crate::ffi::c_void {
349350
ManuallyDrop::new(self).ptr.as_ptr().cast()
@@ -369,6 +370,12 @@ impl<T: 'static> ForeignOwnable for Arc<T> {
369370
// for the lifetime of the returned value.
370371
unsafe { ArcBorrow::new(inner) }
371372
}
373+
374+
unsafe fn borrow_mut<'a>(ptr: *mut crate::ffi::c_void) -> ArcBorrow<'a, T> {
375+
// SAFETY: The safety requirements for `borrow_mut` are a superset of the safety
376+
// requirements for `borrow`.
377+
unsafe { Self::borrow(ptr) }
378+
}
372379
}
373380

374381
impl<T: ?Sized> Deref for Arc<T> {

‎rust/kernel/types.rs

+58-13
Original file line numberDiff line numberDiff line change
@@ -19,26 +19,33 @@ use core::{
1919
/// This trait is meant to be used in cases when Rust objects are stored in C objects and
2020
/// eventually "freed" back to Rust.
2121
pub trait ForeignOwnable: Sized {
22-
/// Type of values borrowed between calls to [`ForeignOwnable::into_foreign`] and
23-
/// [`ForeignOwnable::from_foreign`].
22+
/// Type used to immutably borrow a value that is currently foreign-owned.
2423
type Borrowed<'a>;
2524

25+
/// Type used to mutably borrow a value that is currently foreign-owned.
26+
type BorrowedMut<'a>;
27+
2628
/// Converts a Rust-owned object to a foreign-owned one.
2729
///
2830
/// The foreign representation is a pointer to void. There are no guarantees for this pointer.
2931
/// For example, it might be invalid, dangling or pointing to uninitialized memory. Using it in
30-
/// any way except for [`ForeignOwnable::from_foreign`], [`ForeignOwnable::borrow`],
31-
/// [`ForeignOwnable::try_from_foreign`] can result in undefined behavior.
32+
/// any way except for [`from_foreign`], [`try_from_foreign`], [`borrow`], or [`borrow_mut`] can
33+
/// result in undefined behavior.
34+
///
35+
/// [`from_foreign`]: Self::from_foreign
36+
/// [`try_from_foreign`]: Self::try_from_foreign
37+
/// [`borrow`]: Self::borrow
38+
/// [`borrow_mut`]: Self::borrow_mut
3239
fn into_foreign(self) -> *mut crate::ffi::c_void;
3340

3441
/// Converts a foreign-owned object back to a Rust-owned one.
3542
///
3643
/// # Safety
3744
///
38-
/// `ptr` must have been returned by a previous call to [`ForeignOwnable::into_foreign`] for
39-
/// which a previous matching [`ForeignOwnable::from_foreign`] hasn't been called yet.
40-
/// Additionally, all instances (if any) of values returned by [`ForeignOwnable::borrow`] for
41-
/// this object must have been dropped.
45+
/// The provided pointer must have been returned by a previous call to [`into_foreign`], and it
46+
/// must not be passed to `from_foreign` more than once.
47+
///
48+
/// [`into_foreign`]: Self::into_foreign
4249
unsafe fn from_foreign(ptr: *mut crate::ffi::c_void) -> Self;
4350

4451
/// Tries to convert a foreign-owned object back to a Rust-owned one.
@@ -48,8 +55,9 @@ pub trait ForeignOwnable: Sized {
4855
///
4956
/// # Safety
5057
///
51-
/// `ptr` must either be null or satisfy the safety requirements for
52-
/// [`ForeignOwnable::from_foreign`].
58+
/// `ptr` must either be null or satisfy the safety requirements for [`from_foreign`].
59+
///
60+
/// [`from_foreign`]: Self::from_foreign
5361
unsafe fn try_from_foreign(ptr: *mut crate::ffi::c_void) -> Option<Self> {
5462
if ptr.is_null() {
5563
None
@@ -60,17 +68,53 @@ pub trait ForeignOwnable: Sized {
6068
}
6169
}
6270

63-
/// Borrows a foreign-owned object.
71+
/// Borrows a foreign-owned object immutably.
72+
///
73+
/// This method provides a way to access a foreign-owned value from Rust immutably. It provides
74+
/// you with exactly the same abilities as an `&Self` when the value is Rust-owned.
6475
///
6576
/// # Safety
6677
///
67-
/// `ptr` must have been returned by a previous call to [`ForeignOwnable::into_foreign`] for
68-
/// which a previous matching [`ForeignOwnable::from_foreign`] hasn't been called yet.
78+
/// The provided pointer must have been returned by a previous call to [`into_foreign`], and if
79+
/// the pointer is ever passed to [`from_foreign`], then that call must happen after the end of
80+
/// the lifetime 'a.
81+
///
82+
/// [`into_foreign`]: Self::into_foreign
83+
/// [`from_foreign`]: Self::from_foreign
6984
unsafe fn borrow<'a>(ptr: *mut crate::ffi::c_void) -> Self::Borrowed<'a>;
85+
86+
/// Borrows a foreign-owned object mutably.
87+
///
88+
/// This method provides a way to access a foreign-owned value from Rust mutably. It provides
89+
/// you with exactly the same abilities as an `&mut Self` when the value is Rust-owned, except
90+
/// that the address of the object must not be changed.
91+
///
92+
/// Note that for types like [`Arc`], an `&mut Arc<T>` only gives you immutable access to the
93+
/// inner value, so this method also only provides immutable access in that case.
94+
///
95+
/// In the case of `Box<T>`, this method gives you the ability to modify the inner `T`, but it
96+
/// does not let you change the box itself. That is, you cannot change which allocation the box
97+
/// points at.
98+
///
99+
/// # Safety
100+
///
101+
/// The provided pointer must have been returned by a previous call to [`into_foreign`], and if
102+
/// the pointer is ever passed to [`from_foreign`], then that call must happen after the end of
103+
/// the lifetime 'a.
104+
///
105+
/// The lifetime 'a must not overlap with the lifetime of any other call to [`borrow`] or
106+
/// `borrow_mut` on the same object.
107+
///
108+
/// [`into_foreign`]: Self::into_foreign
109+
/// [`from_foreign`]: Self::from_foreign
110+
/// [`borrow`]: Self::borrow
111+
/// [`Arc`]: crate::sync::Arc
112+
unsafe fn borrow_mut<'a>(ptr: *mut crate::ffi::c_void) -> Self::BorrowedMut<'a>;
70113
}
71114

72115
impl ForeignOwnable for () {
73116
type Borrowed<'a> = ();
117+
type BorrowedMut<'a> = ();
74118

75119
fn into_foreign(self) -> *mut crate::ffi::c_void {
76120
core::ptr::NonNull::dangling().as_ptr()
@@ -79,6 +123,7 @@ impl ForeignOwnable for () {
79123
unsafe fn from_foreign(_: *mut crate::ffi::c_void) -> Self {}
80124

81125
unsafe fn borrow<'a>(_: *mut crate::ffi::c_void) -> Self::Borrowed<'a> {}
126+
unsafe fn borrow_mut<'a>(_: *mut crate::ffi::c_void) -> Self::BorrowedMut<'a> {}
82127
}
83128

84129
/// Runs a cleanup function/closure when dropped.

0 commit comments

Comments
 (0)
Please sign in to comment.