mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-12-27 11:06:41 -05:00
Merge patch series "Rust support for struct iov_iter"
Alice Ryhl <aliceryhl@google.com> says: This series adds support for the `struct iov_iter` type. This type represents an IO buffer for reading or writing, and can be configured for either direction of communication. In Rust, we define separate types for reading and writing. This will ensure that you cannot mix them up and e.g. call copy_from_iter in a read_iter syscall. To use the new abstractions, miscdevices are given new methods read_iter and write_iter that can be used to implement the read/write syscalls on a miscdevice. The miscdevice sample is updated to provide read/write operations. Intended for Greg's miscdevice tree. Link: https://lore.kernel.org/r/20250822-iov-iter-v5-0-6ce4819c2977@google.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
@@ -100,8 +100,9 @@
|
||||
use kernel::{
|
||||
c_str,
|
||||
device::Device,
|
||||
fs::File,
|
||||
fs::{File, Kiocb},
|
||||
ioctl::{_IO, _IOC_SIZE, _IOR, _IOW},
|
||||
iov::{IovIterDest, IovIterSource},
|
||||
miscdevice::{MiscDevice, MiscDeviceOptions, MiscDeviceRegistration},
|
||||
new_mutex,
|
||||
prelude::*,
|
||||
@@ -143,6 +144,7 @@ fn init(_module: &'static ThisModule) -> impl PinInit<Self, Error> {
|
||||
|
||||
struct Inner {
|
||||
value: i32,
|
||||
buffer: KVVec<u8>,
|
||||
}
|
||||
|
||||
#[pin_data(PinnedDrop)]
|
||||
@@ -164,7 +166,10 @@ fn open(_file: &File, misc: &MiscDeviceRegistration<Self>) -> Result<Pin<KBox<Se
|
||||
KBox::try_pin_init(
|
||||
try_pin_init! {
|
||||
RustMiscDevice {
|
||||
inner <- new_mutex!( Inner{ value: 0_i32 } ),
|
||||
inner <- new_mutex!(Inner {
|
||||
value: 0_i32,
|
||||
buffer: KVVec::new(),
|
||||
}),
|
||||
dev: dev,
|
||||
}
|
||||
},
|
||||
@@ -172,6 +177,33 @@ fn open(_file: &File, misc: &MiscDeviceRegistration<Self>) -> Result<Pin<KBox<Se
|
||||
)
|
||||
}
|
||||
|
||||
fn read_iter(mut kiocb: Kiocb<'_, Self::Ptr>, iov: &mut IovIterDest<'_>) -> Result<usize> {
|
||||
let me = kiocb.file();
|
||||
dev_info!(me.dev, "Reading from Rust Misc Device Sample\n");
|
||||
|
||||
let inner = me.inner.lock();
|
||||
// Read the buffer contents, taking the file position into account.
|
||||
let read = iov.simple_read_from_buffer(kiocb.ki_pos_mut(), &inner.buffer)?;
|
||||
|
||||
Ok(read)
|
||||
}
|
||||
|
||||
fn write_iter(mut kiocb: Kiocb<'_, Self::Ptr>, iov: &mut IovIterSource<'_>) -> Result<usize> {
|
||||
let me = kiocb.file();
|
||||
dev_info!(me.dev, "Writing to Rust Misc Device Sample\n");
|
||||
|
||||
let mut inner = me.inner.lock();
|
||||
|
||||
// Replace buffer contents.
|
||||
inner.buffer.clear();
|
||||
let len = iov.copy_from_iter_vec(&mut inner.buffer, GFP_KERNEL)?;
|
||||
|
||||
// Set position to zero so that future `read` calls will see the new contents.
|
||||
*kiocb.ki_pos_mut() = 0;
|
||||
|
||||
Ok(len)
|
||||
}
|
||||
|
||||
fn ioctl(me: Pin<&RustMiscDevice>, _file: &File, cmd: u32, arg: usize) -> Result<isize> {
|
||||
dev_info!(me.dev, "IOCTLing Rust Misc Device Sample\n");
|
||||
|
||||
|
||||
Reference in New Issue
Block a user