From a54505ccd8fcfadf542a9c3bccc6e28423b30123 Mon Sep 17 00:00:00 2001 From: shamb0 Date: Thu, 26 Dec 2024 21:08:42 +0530 Subject: [PATCH] Add callback support to FileDescription - Implementing atomic reads for contiguous buffers Signed-off-by: shamb0 --- src/concurrency/init_once.rs | 4 +- src/concurrency/sync.rs | 13 +- src/concurrency/thread.rs | 7 +- src/shims/files.rs | 85 +---- src/shims/unix/fd.rs | 187 +-------- src/shims/unix/foreign_items.rs | 6 - src/shims/unix/fs.rs | 62 +-- tests/pass-dep/libc/libc-fs.rs | 645 -------------------------------- tests/utils/fs.rs | 47 --- 9 files changed, 45 insertions(+), 1011 deletions(-) diff --git a/src/concurrency/init_once.rs b/src/concurrency/init_once.rs index a2960cb6fb..534f02545b 100644 --- a/src/concurrency/init_once.rs +++ b/src/concurrency/init_once.rs @@ -2,7 +2,7 @@ use std::collections::VecDeque; use rustc_index::Idx; -use super::thread::DyMachineCallback; +use super::thread::DynUnblockCallback; use super::vector_clock::VClock; use crate::*; @@ -35,7 +35,7 @@ pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> { /// Put the thread into the queue waiting for the initialization. #[inline] - fn init_once_enqueue_and_block(&mut self, id: InitOnceId, callback: DyMachineCallback<'tcx>) { + fn init_once_enqueue_and_block(&mut self, id: InitOnceId, callback: DynUnblockCallback<'tcx>) { let this = self.eval_context_mut(); let thread = this.active_thread(); let init_once = &mut this.machine.sync.init_onces[id]; diff --git a/src/concurrency/sync.rs b/src/concurrency/sync.rs index 665279146d..14c72e9398 100644 --- a/src/concurrency/sync.rs +++ b/src/concurrency/sync.rs @@ -428,16 +428,11 @@ pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> { assert!(!this.mutex_is_locked(&mutex_ref)); this.mutex_lock(&mutex_ref); - if let Some((retval, dest)) = retval_dest { - this.write_scalar(retval, &dest)?; - } - - interp_ok(()) - }, - MachineCallbackState::TimedOut => { - panic!("Mutex operation received unexpected timeout state - mutex operations do not support timeouts") - }, + if let Some((retval, dest)) = retval_dest { + this.write_scalar(retval, &dest)?; } + + interp_ok(()) } ), ); diff --git a/src/concurrency/thread.rs b/src/concurrency/thread.rs index 448d26d896..6d22dd8d68 100644 --- a/src/concurrency/thread.rs +++ b/src/concurrency/thread.rs @@ -116,7 +116,7 @@ enum ThreadState<'tcx> { /// The thread is enabled and can be executed. Enabled, /// The thread is blocked on something. - Blocked { reason: BlockReason, timeout: Option, callback: DyMachineCallback<'tcx> }, + Blocked { reason: BlockReason, timeout: Option, callback: DynUnblockCallback<'tcx> }, /// The thread has terminated its execution. We do not delete terminated /// threads (FIXME: why?). Terminated, @@ -609,6 +609,7 @@ impl<'tcx> ThreadManager<'tcx> { if let Some(data_race) = &mut this.machine.data_race { data_race.thread_joined(&this.machine.threads, joined_thread_id); } + interp_ok(()) } ), ); @@ -666,7 +667,7 @@ impl<'tcx> ThreadManager<'tcx> { &mut self, reason: BlockReason, timeout: Option, - callback: DyMachineCallback<'tcx>, + callback: DynUnblockCallback<'tcx>, ) { let state = &mut self.threads[self.active_thread].state; assert!(state.is_enabled()); @@ -988,7 +989,7 @@ pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> { &mut self, reason: BlockReason, timeout: Option<(TimeoutClock, TimeoutAnchor, Duration)>, - callback: DyMachineCallback<'tcx>, + callback: DynUnblockCallback<'tcx>, ) { let this = self.eval_context_mut(); let timeout = timeout.map(|(clock, anchor, duration)| { diff --git a/src/shims/files.rs b/src/shims/files.rs index d1782cf297..eff3a806ff 100644 --- a/src/shims/files.rs +++ b/src/shims/files.rs @@ -147,29 +147,6 @@ pub trait FileDescription: std::fmt::Debug + FileDescriptionExt { throw_unsup_format!("cannot read from {}", self.name()); } - /// Performs an atomic read operation on the file. - /// - /// # Arguments - /// * `self_ref` - Strong reference to file description for lifetime management - /// * `communicate_allowed` - Whether external communication is permitted - /// * `op` - The I/O operation containing buffer and layout information - /// * `dest` - Destination for storing operation results - /// * `ecx` - Mutable reference to interpreter context - /// - /// # Returns - /// * `Ok(())` on successful read - /// * `Err(_)` if read fails or is unsupported - fn read_atomic<'tcx>( - &self, - _self_ref: &FileDescriptionRef, - _communicate_allowed: bool, - _op: &mut IoTransferOperation<'tcx>, - _dest: &MPlaceTy<'tcx>, - _ecx: &mut MiriInterpCx<'tcx>, - ) -> InterpResult<'tcx> { - throw_unsup_format!("cannot read from {}", self.name()); - } - /// Writes as much as possible from the given buffer `ptr`. /// `len` indicates how many bytes we should try to write. /// `dest` is where the return value should be stored: number of bytes written, or `-1` in case of error. @@ -468,7 +445,7 @@ pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> { } /// Represents an atomic I/O operation that handles data transfer between memory regions. -/// Supports both contiguous and scattered memory layouts for efficient I/O operations. +/// Supports contiguous memory layouts for efficient I/O operations. #[derive(Clone)] pub struct IoTransferOperation<'tcx> { /// Intermediate buffer for atomic transfer operations. @@ -491,8 +468,6 @@ pub struct IoTransferOperation<'tcx> { enum IoBufferLayout { /// Single continuous memory region for transfer. Contiguous { address: Pointer }, - /// Multiple discontinuous memory regions. - Scattered { regions: Vec<(Pointer, usize)> }, } impl VisitProvenance for IoTransferOperation<'_> { @@ -513,17 +488,6 @@ impl<'tcx> IoTransferOperation<'tcx> { } } - /// Creates a new I/O operation for scattered memory regions. - pub fn new_scattered(buffers: Vec<(Pointer, usize)>) -> Self { - let total_size = buffers.iter().map(|(_, len)| len).sum(); - IoTransferOperation { - transfer_buffer: vec![0; total_size], - layout: IoBufferLayout::Scattered { regions: buffers }, - total_size, - _phantom: std::marker::PhantomData, - } - } - /// Provides mutable access to the transfer buffer. pub fn buffer_mut(&mut self) -> &mut [u8] { &mut self.transfer_buffer @@ -567,53 +531,6 @@ impl<'tcx> IoTransferOperation<'tcx> { return ecx.set_last_error_and_return(LibcError("EIO"), dest); } } - - IoBufferLayout::Scattered { regions } => { - let mut current_pos = 0; - - for (ptr, len) in regions { - if current_pos >= bytes_processed { - break; - } - - // Calculate copy size with safe arithmetic - let remaining_bytes = bytes_processed - .checked_sub(current_pos) - .expect("current_pos should never exceed bytes_read"); - let copy_size = (*len).min(remaining_bytes); - - // POSIX Compliance: Verify each buffer's accessibility - if ecx - .check_ptr_access( - *ptr, - Size::from_bytes(copy_size), - CheckInAllocMsg::MemoryAccessTest, - ) - .report_err() - .is_err() - { - return ecx.set_last_error_and_return(LibcError("EFAULT"), dest); - } - - let end_pos = current_pos - .checked_add(copy_size) - .expect("end position calculation should not overflow"); - - // Attempt the write operation with proper error handling - if ecx - .write_bytes_ptr( - *ptr, - self.transfer_buffer[current_pos..end_pos].iter().copied(), - ) - .report_err() - .is_err() - { - return ecx.set_last_error_and_return(LibcError("EIO"), dest); - } - - current_pos = end_pos; - } - } } interp_ok(()) diff --git a/src/shims/unix/fd.rs b/src/shims/unix/fd.rs index c09bedb48d..0b59490308 100644 --- a/src/shims/unix/fd.rs +++ b/src/shims/unix/fd.rs @@ -5,10 +5,9 @@ use std::io; use std::io::ErrorKind; use rustc_abi::Size; -use rustc_middle::ty::layout::TyAndLayout; use crate::helpers::check_min_arg_count; -use crate::shims::files::{FileDescription, IoTransferOperation}; +use crate::shims::files::FileDescription; use crate::shims::unix::linux_like::epoll::EpollReadyEvents; use crate::shims::unix::*; use crate::*; @@ -246,10 +245,7 @@ pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> { // `usize::MAX` because it is bounded by the host's `isize`. match offset { - None => { - let mut op = IoTransferOperation::new_contiguous(buf, count); - fd.read_atomic(&fd, communicate, &mut op, dest, this)? - } + None => fd.read(communicate, buf, count, dest, this)?, Some(offset) => { let Ok(offset) = u64::try_from(offset) else { return this.set_last_error_and_return(LibcError("EINVAL"), dest); @@ -260,185 +256,6 @@ pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> { interp_ok(()) } - /// Reads data from a file descriptor into multiple buffers atomically (vectored I/O). - /// - /// This implementation follows POSIX readv() semantics, reading data from the file descriptor - /// specified by `fd_num` into the buffers described by the array of iovec structures pointed - /// to by `iov_ptr`. The `iovcnt` argument specifies the number of iovec structures in the array. - /// - /// # Arguments - /// * `fd_num` - The file descriptor to read from - /// * `iov_ptr` - Pointer to an array of iovec structures - /// * `iovcnt` - Number of iovec structures in the array - /// * `dest` - Destination for storing the number of bytes read - /// - /// # Returns - /// * `Ok(())` - Operation completed successfully, with total bytes read stored in `dest` - /// * `Err(_)` - Operation failed with appropriate errno set - /// - /// # Errors - /// * `EBADF` - `fd_num` is not a valid file descriptor - /// * `EFAULT` - Part of iovec array or buffers point outside accessible address space - /// * `EINVAL` - `iovcnt` is negative or exceeds system limit - /// * `EIO` - I/O error occurred while reading from the file descriptor - /// - /// # POSIX Compliance - /// Implements standard POSIX readv() behavior: - /// * Performs reads atomically with respect to other threads - /// * Returns exact number of bytes read or -1 on error - /// * Handles partial reads and end-of-file conditions - /// * Respects system-imposed limits on total transfer size - fn readv( - &mut self, - fd_num: i32, - iov_ptr: &OpTy<'tcx>, - iovcnt: i32, - dest: &MPlaceTy<'tcx>, - ) -> InterpResult<'tcx> { - let this = self.eval_context_mut(); - - // POSIX Compliance: Must handle negative values (EINVAL). - if iovcnt < 0 { - return this.set_last_error_and_return(LibcError("EINVAL"), dest); - } - - // POSIX Compliance: Must handle zero properly. - if iovcnt == 0 { - return this.write_scalar(Scalar::from_i32(0), dest); - } - - // POSIX Compliance: Check if iovcnt exceeds system limits. - // Most implementations limit this to IOV_MAX - // Common system default - // https://github.com/turbolent/w2c2/blob/d94227c22f8d78a04fbad70fa744481ca4a1912e/examples/clang/sys/include/limits.h#L50 - const IOV_MAX: i32 = 1024; - if iovcnt > IOV_MAX { - trace!("readv: iovcnt exceeds IOV_MAX"); - return this.set_last_error_and_return(LibcError("EINVAL"), dest); - } - - // POSIX Compliance: Validate file descriptor. - let Some(fd) = this.machine.fds.get(fd_num) else { - return this.set_last_error_and_return(LibcError("EBADF"), dest); - }; - - // Convert iovcnt to usize for array indexing. - let iovcnt = usize::try_from(iovcnt).expect("iovcnt exceeds platform size"); - let iovec_layout = this.libc_ty_layout("iovec"); - - // Gather iovec information. - // Pre-allocate vectors for iovec information - let mut iov_info = Vec::with_capacity(iovcnt); - let mut total_size: usize = 0; - let communicate = this.machine.communicate(); - - // POSIX Compliance: Validate each iovec structure. - // Must check for EFAULT (invalid buffer addresses) and EINVAL (invalid length). - for i in 0..iovcnt { - // Calculate offset to current iovec structure. - let offset = iovec_layout - .size - .bytes() - .checked_mul(i as u64) - .expect("iovec array index overflow"); - - // Access current iovec structure. - let current_iov = match this - .deref_pointer_and_offset_vectored( - iov_ptr, - offset, - iovec_layout, - iovcnt, - iovec_layout, - ) - .report_err() - { - Ok(iov) => iov, - Err(_) => { - return this.set_last_error_and_return(LibcError("EFAULT"), dest); - } - }; - - // Extract and validate buffer pointer and length. - let base_field = this.project_field_named(¤t_iov, "iov_base")?; - let base = this.read_pointer(&base_field)?; - - let len_field = this.project_field_named(¤t_iov, "iov_len")?; - let len: usize = usize::try_from(this.read_target_usize(&len_field)?).unwrap(); - - // Validate buffer alignment and accessibility. - if this - .check_ptr_access(base, Size::from_bytes(len), CheckInAllocMsg::MemoryAccessTest) - .report_err() - .is_err() - { - return this.set_last_error_and_return(LibcError("EFAULT"), dest); - } - - // Update total size safely. - total_size = match total_size.checked_add(len) { - Some(new_size) => new_size, - None => { - return this.set_last_error_and_return(LibcError("EINVAL"), dest); - } - }; - - iov_info.push((base, len)); - } - - let mut op = IoTransferOperation::new_scattered(iov_info); - fd.read_atomic(&fd, communicate, &mut op, dest, this)?; - - interp_ok(()) - } - - /// Dereferences a pointer to access an element within a source array, with specialized bounds checking - /// for vectored I/O operations like readv(). - /// - /// This function provides array-aware bounds checking that is specifically designed for situations - /// where we need to access multiple independent memory regions, such as when processing an array - /// of iovec structures. Unlike simple pointer arithmetic bounds checking, this implementation - /// understands and validates array-based access patterns. - fn deref_pointer_and_offset_vectored( - &self, - op: &impl Projectable<'tcx, Provenance>, - offset_bytes: u64, - base_layout: TyAndLayout<'tcx>, - count: usize, - value_layout: TyAndLayout<'tcx>, - ) -> InterpResult<'tcx, MPlaceTy<'tcx>> { - // 1. Validate the iovec array bounds. - let array_size = base_layout - .size - .bytes() - .checked_mul(count as u64) - .ok_or_else(|| err_ub_format!("iovec array size overflow"))?; - - // 2. Check if our offset is within the array. - if offset_bytes >= array_size { - throw_ub_format!( - "{}", - format!( - "iovec array access out of bounds: offset {} in array of size {}", - offset_bytes, array_size - ) - ); - } - - // 3. Ensure the iovec structure we're accessing is fully contained. - if offset_bytes.checked_add(base_layout.size.bytes()).is_none_or(|end| end > array_size) { - throw_ub_format!("iovec structure would extend past array bounds"); - } - - // 4. Proceed with the dereferencing. - let this = self.eval_context_ref(); - let op_place = this.deref_pointer_as(op, base_layout)?; - let offset = Size::from_bytes(offset_bytes); - - let value_place = op_place.offset(offset, value_layout, this)?; - interp_ok(value_place) - } - fn write( &mut self, fd_num: i32, diff --git a/src/shims/unix/foreign_items.rs b/src/shims/unix/foreign_items.rs index 399c7d0433..f47a96b10f 100644 --- a/src/shims/unix/foreign_items.rs +++ b/src/shims/unix/foreign_items.rs @@ -157,12 +157,6 @@ pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> { let count = this.read_target_usize(count)?; this.read(fd, buf, count, None, dest)?; } - "readv" => { - let [fd, iov, iovcnt] = this.check_shim(abi, Conv::C , link_name, args)?; - let fd = this.read_scalar(fd)?.to_i32()?; - let iovcnt = this.read_scalar(iovcnt)?.to_i32()?; - this.readv(fd, iov, iovcnt, dest)?; - } "write" => { let [fd, buf, n] = this.check_shim(abi, Conv::C , link_name, args)?; let fd = this.read_scalar(fd)?.to_i32()?; diff --git a/src/shims/unix/fs.rs b/src/shims/unix/fs.rs index 781fe377bc..8481246f7b 100644 --- a/src/shims/unix/fs.rs +++ b/src/shims/unix/fs.rs @@ -26,6 +26,7 @@ use crate::*; struct FileHandle { file: File, writable: bool, + /// Mutex for synchronizing file access across threads. file_lock: MutexRef, } @@ -37,12 +38,16 @@ impl VisitProvenance for FileHandle { } impl FileHandle { + /// Creates a new FileHandle with specified permissions and synchronization primitive. fn new(file: File, writable: bool, file_lock: MutexRef) -> Self { Self { file, writable, file_lock } } + /// Attempts to create a clone of the file handle while preserving all attributes. + /// + /// # Errors + /// Returns an `InterpResult` error if file handle cloning fails. fn try_clone<'tcx>(&self) -> InterpResult<'tcx, FileHandle> { - // Explicitly handling errors with more context let cloned_file = self .file .try_clone() @@ -55,11 +60,22 @@ impl FileHandle { }) } + /// Performs a synchronized read operation on the file handle. + /// + /// # Arguments + /// * `this` - Interpreter context + /// * `dest` - Destination for the read operation result + /// * `file_handle` - Handle to read from + /// * `weak_fd` - Weak reference to file descriptor for validity checking + /// * `op` - I/O transfer operation specification + /// + /// # Returns + /// InterpResult indicating success or failure of the read operation. fn perform_read<'tcx>( this: &mut MiriInterpCx<'tcx>, dest: MPlaceTy<'tcx>, mut file_handle: FileHandle, - weak_fd: WeakFileDescriptionRef, + weak_fd: WeakFileDescriptionRef, op: std::rc::Rc>>, ) -> InterpResult<'tcx> { this.mutex_lock(&file_handle.file_lock); @@ -101,23 +117,14 @@ impl FileDescription for FileHandle { "file" } - /// Performs an atomic read operation on the file. - /// - /// # Arguments - /// * `self_ref` - Strong reference to file description for lifetime management - /// * `communicate_allowed` - Whether external communication is permitted - /// * `op` - The I/O operation containing buffer and layout information - /// * `dest` - Destination for storing operation results - /// * `ecx` - Mutable reference to interpreter context - /// - /// # Returns - /// * `Ok(())` on successful read - /// * `Err(_)` if read fails or is unsupported - fn read_atomic<'tcx>( - &self, - self_ref: &FileDescriptionRef, + /// Reads as much as possible into the given buffer `ptr`, with proper synchronization and error handling. + /// `len` indicates how many bytes we should try to read. + /// `dest` is where the return value should be stored: number of bytes read, or `-1` in case of error. + fn read<'tcx>( + self: FileDescriptionRef, communicate_allowed: bool, - op: &mut IoTransferOperation<'tcx>, + ptr: Pointer, + len: usize, dest: &MPlaceTy<'tcx>, ecx: &mut MiriInterpCx<'tcx>, ) -> InterpResult<'tcx> { @@ -130,11 +137,12 @@ impl FileDescription for FileHandle { Err(_e) => return this.set_last_error_and_return(LibcError("EIO"), dest), }; - let weak_fd = self_ref.downgrade(); + let op = IoTransferOperation::new_contiguous(ptr, len); + + let weak_fd = FileDescriptionRef::downgrade(&self); let op = std::rc::Rc::new(std::cell::RefCell::new(op.clone())); let dest = dest.clone(); - // Rest of the implementation remains the same if this.mutex_is_locked(&self.file_lock) { this.block_thread( BlockReason::Mutex, @@ -143,18 +151,12 @@ impl FileDescription for FileHandle { @capture<'tcx> { dest: MPlaceTy<'tcx>, clone_file_handle: FileHandle, - weak_fd: WeakFileDescriptionRef, + weak_fd: WeakFileDescriptionRef, op: std::rc::Rc>>, } - @unblock = |this, state| { - match state { - MachineCallbackState::Ready => { - FileHandle::perform_read(this, dest, clone_file_handle, weak_fd, op) - } - MachineCallbackState::TimedOut => { - panic!("Mutex operation received unexpected timeout state") - }, - } + |this, unblock: UnblockKind| { + assert_eq!(unblock, UnblockKind::Ready); + FileHandle::perform_read(this, dest, clone_file_handle, weak_fd, op) } ), ); diff --git a/tests/pass-dep/libc/libc-fs.rs b/tests/pass-dep/libc/libc-fs.rs index 007de62313..f85abe2cc4 100644 --- a/tests/pass-dep/libc/libc-fs.rs +++ b/tests/pass-dep/libc/libc-fs.rs @@ -14,42 +14,6 @@ use std::path::PathBuf; #[path = "../../utils/mod.rs"] mod utils; -/// Platform-specific offset type for file seeking operations. -/// the lseek system call typically expects an off_t type, which can be 64 bits -/// even on some 32-bit systems. -#[cfg(any( - target_os = "illumos", - target_os = "solaris", - target_os = "android", - all(target_os = "linux", target_pointer_width = "64"), - all(target_os = "macos", target_arch = "aarch64") -))] -type LseekOffset = i64; - -#[cfg(any( - target_os = "freebsd", - all(target_os = "macos", not(target_arch = "aarch64")), - all(target_os = "linux", target_pointer_width = "32") -))] -type LseekOffset = i32; - -/// Seeks to a specific position in the file -fn seek(fd: i32, offset: LseekOffset) -> LseekOffset { - let result = unsafe { - // the lseek64 function is not part of the POSIX standard and - // may not be available on all systems. - #[cfg(all(target_os = "linux", target_pointer_width = "64"))] - let result = libc::lseek64(fd, offset.into(), libc::SEEK_SET); - - #[cfg(not(all(target_os = "linux", target_pointer_width = "64")))] - let result = libc::lseek(fd, offset.into(), libc::SEEK_SET); - - result - }; - - LseekOffset::try_from(result).expect("seek operation failed") -} - fn main() { test_dup(); test_dup_stdout_stderr(); @@ -74,12 +38,6 @@ fn main() { test_isatty(); test_read_and_uninit(); test_nofollow_not_symlink(); - test_readv_basic(); - test_readv_large_buffers(); - test_readv_partial_and_eof(); - test_readv_error_conditions(); - test_read_atomic_contiguous(); - test_read_atomic_scattered(); } fn test_file_open_unix_allow_two_args() { @@ -473,606 +431,3 @@ fn test_nofollow_not_symlink() { let ret = unsafe { libc::open(cpath.as_ptr(), libc::O_NOFOLLOW | libc::O_CLOEXEC) }; assert!(ret >= 0); } - -/// Tests basic functionality of the readv() system call by reading a small file -/// with multiple buffers. -/// -/// Verifies that: -/// - File contents are read correctly into provided buffers -/// - The total number of bytes read matches file size -/// - Buffer boundaries are respected -/// - Return values match expected behavior -fn test_readv_basic() { - let bytes = b"abcdefgh"; - let path = utils::prepare_with_content("miri_test_libc_readv.txt", bytes); - - // Convert path to a null-terminated CString. - let name = CString::new(path.into_os_string().into_string().unwrap()).unwrap(); - let name_ptr = name.as_ptr(); - - let mut first_buf = [0u8; 4]; - let mut second_buf = [0u8; 8]; - - unsafe { - // Define iovec structures. - let iov: [libc::iovec; 2] = [ - libc::iovec { - iov_len: first_buf.len() as usize, - iov_base: first_buf.as_mut_ptr() as *mut libc::c_void, - }, - libc::iovec { - iov_len: second_buf.len() as usize, - iov_base: second_buf.as_mut_ptr() as *mut libc::c_void, - }, - ]; - - // Open file. - let fd = libc::open(name_ptr, libc::O_RDONLY); - if fd < 0 { - eprintln!("Failed to open file: {}", Error::last_os_error().to_string()); - return; - } - - // Call readv with proper type conversions. - let iovcnt = libc::c_int::try_from(iov.len()).expect("iovec count too large for platform"); - - // Call readv with proper type handling for the count. - let res = libc::readv(fd, iov.as_ptr() as *const libc::iovec, iovcnt); - - if res < 0 { - eprintln!("Failed to readv: {}", Error::last_os_error()); - libc::close(fd); - return; - } - - // Close the file descriptor. - libc::close(fd); - } - - // Validate buffers. - if first_buf != *b"abcd" { - eprintln!("First buffer mismatch: {:?}", first_buf); - } - - if second_buf != *b"efgh\0\0\0\0" { - eprintln!("Second buffer mismatch: {:?}", second_buf); - } -} - -/// Tests readv() system call with large buffer sizes and pattern verification. -/// Uses multiple buffers (16KB, 16KB, 32KB) to read a 64KB file containing -/// a repeating 'ABCD' pattern with markers at buffer boundaries. -/// -/// Verifies that: -/// - Large file contents are read correctly -/// - Markers at buffer boundaries are preserved -/// - Pattern integrity is maintained between markers -/// - Memory safety with large allocations -/// - Buffer boundary handling for larger sizes -fn test_readv_large_buffers() { - const BUFFER_SIZE_1: usize = 16384; // 16KB - const BUFFER_SIZE_2: usize = 16384; // 16KB - const BUFFER_SIZE_3: usize = 32768; // 32KB - - // Define our buffer sizes - let buffer_sizes = &[ - BUFFER_SIZE_1, // 16KB - BUFFER_SIZE_2, // 16KB - BUFFER_SIZE_3, // 32KB - ]; - - // Create large test file with patterns and markers. - // Generate pattern with awareness of buffer boundaries. - let large_content = utils::generate_test_pattern(buffer_sizes); - - let path = utils::prepare_with_content("large_readv_test.txt", &large_content); - - // Create buffers based on our defined sizes. - let mut buffers: Vec> = buffer_sizes.iter().map(|&size| vec![0u8; size]).collect(); - - // Convert path to CString for libc interface. - let path_cstr = CString::new(path.into_os_string().into_string().unwrap()).unwrap(); - - let bytes_read: usize = unsafe { - let fd = libc::open(path_cstr.as_ptr(), libc::O_RDONLY); - assert!(fd > 0, "Failed to open test file"); - - // Create iovec array using our buffers. - let iov = buffers - .iter_mut() - .map(|buf| { - libc::iovec { iov_base: buf.as_mut_ptr() as *mut libc::c_void, iov_len: buf.len() } - }) - .collect::>(); - - // Perform readv operation. - let read_result = libc::readv(fd, iov.as_ptr(), iov.len() as i32); - - libc::close(fd); - read_result.try_into().unwrap() - }; - - // Verify total bytes read. - let expected_total: usize = buffer_sizes.iter().sum(); - assert_eq!( - bytes_read, expected_total, - "Unexpected bytes read. Expected {}, got {}", - expected_total, bytes_read - ); - - // Verify markers in each buffer with correct positioning. - let mut current_pos = 0; - for (i, buf) in buffers.iter().enumerate() { - let marker = format!("##MARKER{}##", i + 1); - let marker_len = marker.len(); - - // Calculate correct position for this buffer - let buffer_size = buf.len(); - let marker_pos = buffer_size - marker_len; - - // Read the exact number of bytes needed for the marker. - let content = std::str::from_utf8(&buf[marker_pos..marker_pos + marker_len]) - .unwrap_or("Invalid UTF-8"); - - assert_eq!( - content, - marker, - "Marker {} mismatch at position {}. Expected '{}', found '{}'", - i + 1, - current_pos + marker_pos, - marker, - content - ); - - // Update position for next buffer - current_pos += buffer_size; - } - - // Helper function to verify the repeating ABCD pattern. - let verify_pattern = |buf: &[u8], start: usize, end: usize, buffer_num: usize| { - // Safety check for range validity - if start >= end || end > buf.len() { - println!( - "Invalid range for buffer {}: start={}, end={}, len={}", - buffer_num, - start, - end, - buf.len() - ); - return false; - } - - let chunk = &buf[start..end]; - - // Calculate the pattern offset for alignment. - let pattern_offset = start % 4; - let expected_pattern = [b'A', b'B', b'C', b'D']; - - // Verify each byte against the expected pattern at the correct offset. - chunk.iter().enumerate().all(|(i, &byte)| { - let expected = expected_pattern[(i + pattern_offset) % 4]; - if byte != expected { - println!( - "Mismatch at position {}: expected {}, found {}", - start + i, - expected as char, - byte as char - ); - false - } else { - true - } - }) - }; - - // Adjust verification ranges and pattern alignment. - for (i, buf) in buffers.iter().enumerate() { - let buffer_num = i + 1; - let buffer_size = buf.len(); - let marker_len = 11; - - // Calculate correct start position based on marker alignment. - let start = if buffer_num == 1 { 0 } else { marker_len }; - let end = buffer_size - marker_len; - - assert!( - verify_pattern(buf, start, end, buffer_num), - "Pattern corruption detected in buffer {}. Expected aligned 'ABCD' pattern \ - in range {}..{}", - buffer_num, - start, - end - ); - } -} - -/// Tests readv() system call behavior with EOF conditions and partial reads. -/// Uses a test file smaller than total buffer size to verify correct handling -/// of file boundaries and partial data transfers. -/// -/// Verifies that: -/// - Partial reads near EOF work correctly -/// - Reading exactly at EOF returns 0 -/// - Buffer contents match expected data -/// - Total bytes read matches available data -/// - Remaining buffer space is unmodified -fn test_readv_partial_and_eof() { - // Let's create a file smaller than our total buffer sizes. - // We'll use a structured pattern to make validation easier. - let test_data = b"HEADER_DATA_SECTION_ONE_DATA_SECTION_TWO_END"; // 41 bytes - let path = utils::prepare_with_content("partial_read_test.txt", test_data); - - // Test Case 1: Normal buffers larger than file size. - { - let mut first_buf = vec![0u8; 20]; // Should be filled completely - let mut second_buf = vec![0u8; 20]; // Should be filled completely - let mut third_buf = vec![0u8; 20]; // Should be partially filled - - let path_cstr = CString::new(path.to_str().unwrap()).unwrap(); - - let bytes_read: usize = unsafe { - let fd = libc::open(path_cstr.as_ptr(), libc::O_RDONLY); - assert!(fd > 0, "Failed to open test file"); - - let iov = [ - libc::iovec { - iov_base: first_buf.as_mut_ptr() as *mut libc::c_void, - iov_len: first_buf.len(), - }, - libc::iovec { - iov_base: second_buf.as_mut_ptr() as *mut libc::c_void, - iov_len: second_buf.len(), - }, - libc::iovec { - iov_base: third_buf.as_mut_ptr() as *mut libc::c_void, - iov_len: third_buf.len(), - }, - ]; - - let result = libc::readv(fd, iov.as_ptr(), iov.len() as i32); - libc::close(fd); - result.try_into().unwrap() - }; - - // Verify total bytes read matches file size. - assert_eq!( - bytes_read, - test_data.len(), - "Expected {} bytes read, got {}", - test_data.len(), - bytes_read - ); - - // Verify buffer contents - assert_eq!(&first_buf[..20], &test_data[..20], "First buffer content mismatch"); - assert_eq!(&second_buf[..20], &test_data[20..40], "Second buffer content mismatch"); - assert_eq!(&third_buf[..1], &test_data[40..41], "Third buffer partial content mismatch"); - } - - // Test Case 2: Reading from an offset near EOF. - { - let mut first_buf = vec![0u8; 10]; - let mut second_buf = vec![0u8; 10]; - - let path_cstr = CString::new(path.to_str().unwrap()).unwrap(); - - let bytes_read: usize = unsafe { - let fd = libc::open(path_cstr.as_ptr(), libc::O_RDONLY); - assert!(fd > 0, "Failed to open test file"); - - // Seek to near end of file - // Use the platform-specific offset type directly - let offset = LseekOffset::try_from(test_data.len() - 15).unwrap(); - let seek_result = seek(fd, offset); - - // Compare using the same types - assert_eq!(LseekOffset::try_from(seek_result).unwrap(), offset); - - let iov = [ - libc::iovec { - iov_base: first_buf.as_mut_ptr() as *mut libc::c_void, - iov_len: first_buf.len(), - }, - libc::iovec { - iov_base: second_buf.as_mut_ptr() as *mut libc::c_void, - iov_len: second_buf.len(), - }, - ]; - - let result = libc::readv(fd, iov.as_ptr(), iov.len() as i32); - libc::close(fd); - result.try_into().unwrap() - }; - - // Should read remaining 15 bytes - assert_eq!(bytes_read, 15, "Expected 15 bytes read from offset, got {}", bytes_read); - } - - // Test Case 3: Reading at EOF. - { - let mut buf = vec![0u8; 10]; - - let path_cstr = CString::new(path.to_str().unwrap()).unwrap(); - - let bytes_read: usize = unsafe { - let fd = libc::open(path_cstr.as_ptr(), libc::O_RDONLY); - assert!(fd > 0, "Failed to open test file"); - - // Seek to EOF - // Cast the offset to the appropriate type for the platform - let offset = LseekOffset::try_from(test_data.len()).unwrap(); - let seek_result = seek(fd, offset); - assert_eq!( - LseekOffset::try_from(seek_result).unwrap(), - LseekOffset::try_from(test_data.len()).unwrap() - ); - - let iov = [libc::iovec { - iov_base: buf.as_mut_ptr() as *mut libc::c_void, - iov_len: buf.len(), - }]; - - let result = libc::readv(fd, iov.as_ptr(), iov.len() as i32); - libc::close(fd); - result.try_into().unwrap() - }; - - // Should read 0 bytes at EOF - assert_eq!(bytes_read, 0, "Expected 0 bytes read at EOF, got {}", bytes_read); - } - - // Test Case 4: Small buffers with exact boundaries. - { - let mut first_buf = vec![0u8; 7]; // "HEADER_" - let mut second_buf = vec![0u8; 5]; // "DATA_" - let mut third_buf = vec![0u8; 7]; // "SECTION" - - let path_cstr = CString::new(path.to_str().unwrap()).unwrap(); - - let bytes_read: usize = unsafe { - let fd = libc::open(path_cstr.as_ptr(), libc::O_RDONLY); - assert!(fd > 0, "Failed to open test file"); - - let iov = [ - libc::iovec { - iov_base: first_buf.as_mut_ptr() as *mut libc::c_void, - iov_len: first_buf.len(), - }, - libc::iovec { - iov_base: second_buf.as_mut_ptr() as *mut libc::c_void, - iov_len: second_buf.len(), - }, - libc::iovec { - iov_base: third_buf.as_mut_ptr() as *mut libc::c_void, - iov_len: third_buf.len(), - }, - ]; - - let result = libc::readv(fd, iov.as_ptr(), iov.len() as i32); - libc::close(fd); - result.try_into().unwrap() - }; - - // Verify exact buffer fills. - assert_eq!( - bytes_read, 19, - "Expected 19 bytes read for exact boundaries, got {}", - bytes_read - ); - assert_eq!(&first_buf, b"HEADER_", "First buffer exact content mismatch"); - assert_eq!(&second_buf, b"DATA_", "Second buffer exact content mismatch"); - assert_eq!(&third_buf[..7], b"SECTION", "Third buffer exact content mismatch"); - } -} - -/// Tests error handling conditions of the readv() system call. -/// Verifies that the implementation properly handles various error scenarios -/// including invalid file descriptors, -/// -/// Test coverage includes: -/// - Invalid file descriptor scenarios -fn test_readv_error_conditions() { - #[cfg(any(target_os = "illumos", target_os = "solaris"))] - use libc::___errno as __errno_location; - #[cfg(target_os = "android")] - use libc::__errno as __errno_location; - #[cfg(target_os = "linux")] - use libc::__errno_location; - #[cfg(any(target_os = "freebsd", target_os = "macos"))] - use libc::__error as __errno_location; - - // Test Case 1: Invalid File Descriptor Scenarios. - { - let mut buffer = vec![0u8; 10]; - - // Create a single valid iovec structure for testing. - let iov = [libc::iovec { - iov_base: buffer.as_mut_ptr() as *mut libc::c_void, - iov_len: buffer.len(), - }]; - - unsafe { - // Test with negative file descriptor. - let result = libc::readv(-1, iov.as_ptr(), 1); - assert_eq!(result, -1, "Expected error for negative file descriptor"); - assert_eq!( - *__errno_location(), - libc::EBADF, - "Expected EBADF for negative file descriptor" - ); - - // Test with unopened but potentially valid fd number. - let result = libc::readv(999999, iov.as_ptr(), 1); - assert_eq!(result, -1, "Expected error for invalid file descriptor"); - assert_eq!( - *__errno_location(), - libc::EBADF, - "Expected EBADF for invalid file descriptor" - ); - } - } -} - -/// Tests concurrent atomic reads with a contiguous buffer. -/// Validates that multiple threads can safely read from the same file -/// using a single continuous memory region for each read operation. -fn test_read_atomic_contiguous() { - // Create test file with known content - let content = b"The quick brown fox jumps over the lazy dog"; - let path = utils::prepare_with_content("miri_test_atomic_read_contiguous.txt", content); - - // Convert path to a string that can be safely cloned for each thread - let path_string = path.into_os_string().into_string().unwrap(); - - // Spawn multiple threads to read concurrently - let mut handles = Vec::new(); - let thread_count = 4; - let read_size = 8; // Each thread reads 8 bytes - - for thread_id in 0..thread_count { - // Clone the path string for this thread - let thread_path = path_string.clone(); - - let handle = std::thread::spawn(move || { - // Create CString inside the thread - let name = CString::new(thread_path).unwrap(); - let name_ptr = name.as_ptr(); - - unsafe { - // Open file descriptor for this thread - let fd = libc::open(name_ptr, libc::O_RDONLY); - assert!(fd >= 0, "Failed to open file in thread {}", thread_id); - - // Seek to thread-specific offset - let offset = thread_id * read_size; - assert_eq!( - libc::lseek(fd, offset as i64, libc::SEEK_SET), - offset as i64, - "Failed to seek in thread {}", - thread_id - ); - - // Perform atomic read - let mut buffer = vec![0u8; read_size]; - let bytes_read = - libc::read(fd, buffer.as_mut_ptr() as *mut libc::c_void, read_size as usize); - - assert_eq!( - bytes_read as usize, read_size, - "Incorrect read size in thread {}", - thread_id - ); - - // Verify read content - assert_eq!( - &buffer, - &content[offset..offset + read_size], - "Incorrect data read in thread {}", - thread_id - ); - - libc::close(fd); - } - }); - - handles.push(handle); - } - - // Wait for all threads to complete - for handle in handles { - handle.join().unwrap(); - } -} - -/// Tests concurrent atomic reads with scattered buffers. -/// Validates that multiple threads can safely read from the same file -/// using multiple discontinuous memory regions for each read operation. -fn test_read_atomic_scattered() { - // Create test file with repeating pattern for easy verification - let pattern = b"ABCDEFGH"; - let repeat_count = 4; - let test_content: Vec = - pattern.iter().cycle().take(pattern.len() * repeat_count).copied().collect(); - let path = utils::prepare_with_content("miri_test_atomic_read_scattered.txt", &test_content); - - // Convert path to a string that can be safely cloned for each thread - let path_string = path.into_os_string().into_string().unwrap(); - - // Spawn threads for concurrent scattered reads - let mut handles = Vec::new(); - let thread_count = 2; - - for thread_id in 0..thread_count { - // Clone the path string for this thread - let thread_path = path_string.clone(); - let content = test_content.clone(); - - let handle = std::thread::spawn(move || { - // Convert path to C string for libc - let name = CString::new(thread_path).unwrap(); - let name_ptr = name.as_ptr(); - - unsafe { - // Open file descriptor - let fd = libc::open(name_ptr, libc::O_RDONLY); - assert!(fd >= 0, "Failed to open file in thread {}", thread_id); - - // Create scattered buffers: two 4-byte regions - let mut buffer1 = vec![0u8; 4]; - let mut buffer2 = vec![0u8; 4]; - - // Set up iovec structures - let iov = [ - libc::iovec { - iov_base: buffer1.as_mut_ptr() as *mut libc::c_void, - iov_len: buffer1.len(), - }, - libc::iovec { - iov_base: buffer2.as_mut_ptr() as *mut libc::c_void, - iov_len: buffer2.len(), - }, - ]; - - // Seek to thread-specific offset - let offset = thread_id * 8; // Each thread reads 8 bytes total - assert_eq!( - libc::lseek(fd, offset as i64, libc::SEEK_SET), - offset as i64, - "Failed to seek in thread {}", - thread_id - ); - - // Perform scattered read - let bytes_read = libc::readv(fd, iov.as_ptr(), 2); - assert_eq!( - bytes_read as usize, 8, - "Incorrect scattered read size in thread {}", - thread_id - ); - - // Verify read content - assert_eq!( - &buffer1, - &content[offset..offset + 4], - "Incorrect data in first buffer of thread {}", - thread_id - ); - assert_eq!( - &buffer2, - &content[offset + 4..offset + 8], - "Incorrect data in second buffer of thread {}", - thread_id - ); - - libc::close(fd); - } - }); - - handles.push(handle); - } - - // Wait for all threads to complete - for handle in handles { - handle.join().unwrap(); - } -} diff --git a/tests/utils/fs.rs b/tests/utils/fs.rs index fcaae1d8e8..7340908626 100644 --- a/tests/utils/fs.rs +++ b/tests/utils/fs.rs @@ -51,50 +51,3 @@ pub fn prepare_dir(dirname: &str) -> PathBuf { fs::remove_dir_all(&path).ok(); path } - -/// Generates a test pattern with markers placed at buffer boundaries -/// -/// Arguments: -/// * `buffer_sizes` - An array of buffer sizes that will be used in the readv operation -/// -/// Returns: -/// * A vector containing the test pattern with markers placed at buffer boundaries -/// -/// The function creates a pattern by: -/// 1. Filling the content with repeating "ABCD" sequences -/// 2. Placing markers at each buffer boundary -/// 3. Adding an end pattern to detect overruns -pub fn generate_test_pattern(buffer_sizes: &[usize]) -> Vec { - // Calculate total size needed for all buffers. - let total_size: usize = buffer_sizes.iter().sum(); - - // Create our base content vector. - let mut content = Vec::with_capacity(total_size); - - // Fill with repeating ABCD pattern. - let base_pattern = b"ABCD"; - while content.len() < total_size { - content.extend_from_slice(base_pattern); - } - content.truncate(total_size); - - // Calculate marker positions at buffer boundaries. - // We'll accumulate sizes to find boundary positions. - // Calculate correct marker positions based on cumulative buffer boundaries. - let mut cumulative_position = 0; - for (i, &buffer_size) in buffer_sizes.iter().enumerate() { - let marker = format!("##MARKER{}##", i + 1).into_bytes(); - let marker_len = marker.len(); - - // Position marker relative to the current buffer's end - let marker_position = cumulative_position + buffer_size - marker_len; - - if marker_position + marker_len <= total_size { - content[marker_position..marker_position + marker_len].copy_from_slice(&marker); - } - - cumulative_position += buffer_size; - } - - content -}