Skip to content

Commit 47cb6bf

Browse files
dingxiangfei2009ojeda
authored andcommitted
rust: use derive(CoercePointee) on rustc >= 1.84.0
The `kernel` crate relies on both `coerce_unsized` and `dispatch_from_dyn` unstable features. Alice Ryhl has proposed [1] the introduction of the unstable macro `SmartPointer` to reduce such dependence, along with a RFC patch [2]. Since Rust 1.81.0 this macro, later renamed to `CoercePointee` in Rust 1.84.0 [3], has been fully implemented with the naming discussion resolved. This feature is now on track to stabilization in the language. In order to do so, we shall start using this macro in the `kernel` crate to prove the functionality and utility of the macro as the justification of its stabilization. This patch makes this switch in such a way that the crate remains backward compatible with older Rust compiler versions, via the new Kconfig option `RUSTC_HAS_COERCE_POINTEE`. A minimal demonstration example is added to the `samples/rust/rust_print_main.rs` module. Link: https://rust-lang.github.io/rfcs/3621-derive-smart-pointer.html [1] Link: https://lore.kernel.org/all/[email protected]/ [2] Link: rust-lang/rust#131284 [3] Signed-off-by: Xiangfei Ding <[email protected]> Reviewed-by: Fiona Behrens <[email protected]> Reviewed-by: Alice Ryhl <[email protected]> Link: https://lore.kernel.org/r/[email protected] [ Fixed version to 1.84. Renamed option to `RUSTC_HAS_COERCE_POINTEE` to match `CC_HAS_*` ones. Moved up new config option, closer to the `CC_HAS_*` ones. Simplified Kconfig line. Fixed typos and slightly reworded example and commit. Added Link to PR. - Miguel ] Signed-off-by: Miguel Ojeda <[email protected]>
1 parent 91da5a2 commit 47cb6bf

File tree

6 files changed

+43
-11
lines changed

6 files changed

+43
-11
lines changed

init/Kconfig

+3
Original file line numberDiff line numberDiff line change
@@ -129,6 +129,9 @@ config CC_HAS_COUNTED_BY
129129
# https://github.com/llvm/llvm-project/pull/112636
130130
depends on !(CC_IS_CLANG && CLANG_VERSION < 190103)
131131

132+
config RUSTC_HAS_COERCE_POINTEE
133+
def_bool RUSTC_VERSION >= 108400
134+
132135
config PAHOLE_VERSION
133136
int
134137
default $(shell,$(srctree)/scripts/pahole-version.sh $(PAHOLE))

rust/kernel/alloc.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -123,7 +123,7 @@ pub mod flags {
123123
/// [`Allocator`] is designed to be implemented as a ZST; [`Allocator`] functions do not operate on
124124
/// an object instance.
125125
///
126-
/// In order to be able to support `#[derive(SmartPointer)]` later on, we need to avoid a design
126+
/// In order to be able to support `#[derive(CoercePointee)]` later on, we need to avoid a design
127127
/// that requires an `Allocator` to be instantiated, hence its functions must not contain any kind
128128
/// of `self` parameter.
129129
///

rust/kernel/lib.rs

+4-3
Original file line numberDiff line numberDiff line change
@@ -13,11 +13,12 @@
1313
1414
#![no_std]
1515
#![feature(arbitrary_self_types)]
16-
#![feature(coerce_unsized)]
17-
#![feature(dispatch_from_dyn)]
16+
#![cfg_attr(CONFIG_RUSTC_HAS_COERCE_POINTEE, feature(derive_coerce_pointee))]
17+
#![cfg_attr(not(CONFIG_RUSTC_HAS_COERCE_POINTEE), feature(coerce_unsized))]
18+
#![cfg_attr(not(CONFIG_RUSTC_HAS_COERCE_POINTEE), feature(dispatch_from_dyn))]
19+
#![cfg_attr(not(CONFIG_RUSTC_HAS_COERCE_POINTEE), feature(unsize))]
1820
#![feature(inline_const)]
1921
#![feature(lint_reasons)]
20-
#![feature(unsize)]
2122

2223
// Ensure conditional compilation based on the kernel configuration works;
2324
// otherwise we may silently break things like initcall handling.

rust/kernel/list/arc.rs

+6-3
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
use crate::alloc::{AllocError, Flags};
88
use crate::prelude::*;
99
use crate::sync::{Arc, ArcBorrow, UniqueArc};
10-
use core::marker::{PhantomPinned, Unsize};
10+
use core::marker::PhantomPinned;
1111
use core::ops::Deref;
1212
use core::pin::Pin;
1313
use core::sync::atomic::{AtomicBool, Ordering};
@@ -159,6 +159,7 @@ pub use impl_list_arc_safe;
159159
///
160160
/// [`List`]: crate::list::List
161161
#[repr(transparent)]
162+
#[cfg_attr(CONFIG_RUSTC_HAS_COERCE_POINTEE, derive(core::marker::CoercePointee))]
162163
pub struct ListArc<T, const ID: u64 = 0>
163164
where
164165
T: ListArcSafe<ID> + ?Sized,
@@ -443,18 +444,20 @@ where
443444

444445
// This is to allow coercion from `ListArc<T>` to `ListArc<U>` if `T` can be converted to the
445446
// dynamically-sized type (DST) `U`.
447+
#[cfg(not(CONFIG_RUSTC_HAS_COERCE_POINTEE))]
446448
impl<T, U, const ID: u64> core::ops::CoerceUnsized<ListArc<U, ID>> for ListArc<T, ID>
447449
where
448-
T: ListArcSafe<ID> + Unsize<U> + ?Sized,
450+
T: ListArcSafe<ID> + core::marker::Unsize<U> + ?Sized,
449451
U: ListArcSafe<ID> + ?Sized,
450452
{
451453
}
452454

453455
// This is to allow `ListArc<U>` to be dispatched on when `ListArc<T>` can be coerced into
454456
// `ListArc<U>`.
457+
#[cfg(not(CONFIG_RUSTC_HAS_COERCE_POINTEE))]
455458
impl<T, U, const ID: u64> core::ops::DispatchFromDyn<ListArc<U, ID>> for ListArc<T, ID>
456459
where
457-
T: ListArcSafe<ID> + Unsize<U> + ?Sized,
460+
T: ListArcSafe<ID> + core::marker::Unsize<U> + ?Sized,
458461
U: ListArcSafe<ID> + ?Sized,
459462
{
460463
}

rust/kernel/sync/arc.rs

+11-4
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ use crate::{
2626
use core::{
2727
alloc::Layout,
2828
fmt,
29-
marker::{PhantomData, Unsize},
29+
marker::PhantomData,
3030
mem::{ManuallyDrop, MaybeUninit},
3131
ops::{Deref, DerefMut},
3232
pin::Pin,
@@ -125,6 +125,8 @@ mod std_vendor;
125125
/// let coerced: Arc<dyn MyTrait> = obj;
126126
/// # Ok::<(), Error>(())
127127
/// ```
128+
#[repr(transparent)]
129+
#[cfg_attr(CONFIG_RUSTC_HAS_COERCE_POINTEE, derive(core::marker::CoercePointee))]
128130
pub struct Arc<T: ?Sized> {
129131
ptr: NonNull<ArcInner<T>>,
130132
// NB: this informs dropck that objects of type `ArcInner<T>` may be used in `<Arc<T> as
@@ -180,10 +182,12 @@ impl<T: ?Sized> ArcInner<T> {
180182

181183
// This is to allow coercion from `Arc<T>` to `Arc<U>` if `T` can be converted to the
182184
// dynamically-sized type (DST) `U`.
183-
impl<T: ?Sized + Unsize<U>, U: ?Sized> core::ops::CoerceUnsized<Arc<U>> for Arc<T> {}
185+
#[cfg(not(CONFIG_RUSTC_HAS_COERCE_POINTEE))]
186+
impl<T: ?Sized + core::marker::Unsize<U>, U: ?Sized> core::ops::CoerceUnsized<Arc<U>> for Arc<T> {}
184187

185188
// This is to allow `Arc<U>` to be dispatched on when `Arc<T>` can be coerced into `Arc<U>`.
186-
impl<T: ?Sized + Unsize<U>, U: ?Sized> core::ops::DispatchFromDyn<Arc<U>> for Arc<T> {}
189+
#[cfg(not(CONFIG_RUSTC_HAS_COERCE_POINTEE))]
190+
impl<T: ?Sized + core::marker::Unsize<U>, U: ?Sized> core::ops::DispatchFromDyn<Arc<U>> for Arc<T> {}
187191

188192
// SAFETY: It is safe to send `Arc<T>` to another thread when the underlying `T` is `Sync` because
189193
// it effectively means sharing `&T` (which is safe because `T` is `Sync`); additionally, it needs
@@ -479,14 +483,17 @@ impl<T: ?Sized> From<Pin<UniqueArc<T>>> for Arc<T> {
479483
/// obj.as_arc_borrow().use_reference();
480484
/// # Ok::<(), Error>(())
481485
/// ```
486+
#[repr(transparent)]
487+
#[cfg_attr(CONFIG_RUSTC_HAS_COERCE_POINTEE, derive(core::marker::CoercePointee))]
482488
pub struct ArcBorrow<'a, T: ?Sized + 'a> {
483489
inner: NonNull<ArcInner<T>>,
484490
_p: PhantomData<&'a ()>,
485491
}
486492

487493
// This is to allow `ArcBorrow<U>` to be dispatched on when `ArcBorrow<T>` can be coerced into
488494
// `ArcBorrow<U>`.
489-
impl<T: ?Sized + Unsize<U>, U: ?Sized> core::ops::DispatchFromDyn<ArcBorrow<'_, U>>
495+
#[cfg(not(CONFIG_RUSTC_HAS_COERCE_POINTEE))]
496+
impl<T: ?Sized + core::marker::Unsize<U>, U: ?Sized> core::ops::DispatchFromDyn<ArcBorrow<'_, U>>
490497
for ArcBorrow<'_, T>
491498
{
492499
}

samples/rust/rust_print_main.rs

+18
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,24 @@ fn arc_print() -> Result {
3434
// Uses `dbg` to print, will move `c` (for temporary debugging purposes).
3535
dbg!(c);
3636

37+
{
38+
// `Arc` can be used to delegate dynamic dispatch and the following is an example.
39+
// Both `i32` and `&str` implement `Display`. This enables us to express a unified
40+
// behaviour, contract or protocol on both `i32` and `&str` into a single `Arc` of
41+
// type `Arc<dyn Display>`.
42+
43+
use core::fmt::Display;
44+
fn arc_dyn_print(arc: &Arc<dyn Display>) {
45+
pr_info!("Arc<dyn Display> says {arc}");
46+
}
47+
48+
let a_i32_display: Arc<dyn Display> = Arc::new(42i32, GFP_KERNEL)?;
49+
let a_str_display: Arc<dyn Display> = a.clone();
50+
51+
arc_dyn_print(&a_i32_display);
52+
arc_dyn_print(&a_str_display);
53+
}
54+
3755
// Pretty-prints the debug formatting with lower-case hexadecimal integers.
3856
pr_info!("{:#x?}", a);
3957

0 commit comments

Comments
 (0)