Skip to main content

starnix_core/vfs/buffers/
io_buffers.rs

1// Copyright 2023 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5use crate::mm::{
6    MemoryAccessorExt, NumberOfElementsRead, TaskMemoryAccessor, UNIFIED_ASPACES_ENABLED,
7    read_to_array, read_to_object_as_bytes, read_to_vec,
8};
9use crate::task::{CurrentTask, Task};
10use smallvec::{SmallVec, smallvec};
11use starnix_types::user_buffer::{UserBuffer, UserBuffers};
12use starnix_uapi::errors::{ENOTSUP, Errno};
13use starnix_uapi::user_address::UserAddress;
14use starnix_uapi::{errno, error};
15use std::mem::MaybeUninit;
16use std::ops::{Deref, DerefMut};
17use zerocopy::FromBytes;
18
19/// The callback for `OutputBuffer::write_each`. The callback is passed the buffers to write to in
20/// order, and must return for each, how many bytes has been written.
21pub type OutputBufferCallback<'a> = dyn FnMut(&mut [MaybeUninit<u8>]) -> Result<usize, Errno> + 'a;
22
23fn slice_to_maybe_uninit(buffer: &[u8]) -> &[MaybeUninit<u8>] {
24    // SAFETY: &[u8] and &[MaybeUninit<u8>] have the same layout.
25    unsafe { std::slice::from_raw_parts(buffer.as_ptr() as *const MaybeUninit<u8>, buffer.len()) }
26}
27
28pub trait Iovec: Sized {
29    fn create(buffer: &UserBuffer) -> Self;
30}
31
32impl Iovec for syncio::zxio::iovec {
33    fn create(buffer: &UserBuffer) -> Self {
34        Self { iov_base: buffer.address.ptr() as *mut starnix_uapi::c_void, iov_len: buffer.length }
35    }
36}
37
38impl Iovec for syncio::zxio::zx_iovec {
39    fn create(buffer: &UserBuffer) -> Self {
40        Self { buffer: buffer.address.ptr() as *mut starnix_uapi::c_void, capacity: buffer.length }
41    }
42}
43
44impl Iovec for zx::sys::zx_iovec_t {
45    fn create(buffer: &UserBuffer) -> Self {
46        Self { buffer: buffer.address.ptr() as *const u8, capacity: buffer.length }
47    }
48}
49
50const IOVECS_IN_HEAP_THRESHOLD: usize = 5;
51
52/// Provides access to a slice of iovecs while retaining some reference.
53pub struct IovecsRef<'a, I: Sized> {
54    iovecs: SmallVec<[I; IOVECS_IN_HEAP_THRESHOLD]>,
55    _marker: std::marker::PhantomData<&'a I>,
56}
57
58impl<'a, I: Iovec> IovecsRef<'a, I> {
59    /// Returns the list of iovecs backing the buffer.
60    ///
61    /// Note that we use `IovecsRef<'_>` so that while `IovecsRef` is held,
62    /// no other methods may be called on the `Buffer` since `IovecsRef`
63    /// holds onto the mutable reference for the `Buffer`.
64    fn new<B: Buffer + ?Sized>(buf: &'a mut B) -> Result<Self, Errno> {
65        let mut iovecs = SmallVec::with_capacity(buf.segments_count()?);
66        buf.peek_each_segment(&mut |buffer| iovecs.push(I::create(buffer)))?;
67        Ok(IovecsRef { iovecs, _marker: Default::default() })
68    }
69}
70
71impl<I> Deref for IovecsRef<'_, I> {
72    type Target = [I];
73    fn deref(&self) -> &Self::Target {
74        &self.iovecs
75    }
76}
77
78impl<I> DerefMut for IovecsRef<'_, I> {
79    fn deref_mut(&mut self) -> &mut Self::Target {
80        &mut self.iovecs
81    }
82}
83
84pub type PeekBufferSegmentsCallback<'a> = dyn FnMut(&UserBuffer) + 'a;
85
86/// A buffer.
87///
88/// Provides the common implementations for input and output buffers.
89pub trait Buffer: std::fmt::Debug {
90    /// Returns the number of segments, if the buffer supports I/O directly
91    /// to/from individual segments.
92    fn segments_count(&self) -> Result<usize, Errno>;
93
94    /// Calls the callback with each segment backing this buffer.
95    ///
96    /// Each segment can be read from (if this is an `InputBuffer`) or written to (if this is an
97    /// `OutputBuffer`) using either user copy routines or Zircon system calls.  If this is an
98    /// `OutputBuffer`, any damage caused by bad addresses will be restricted to user addresses.
99    fn peek_each_segment(
100        &mut self,
101        callback: &mut PeekBufferSegmentsCallback<'_>,
102    ) -> Result<(), Errno>;
103
104    /// Returns all the segments backing this `Buffer`.
105    ///
106    /// Note that we use `IovecsRef<'_>` so that while `IovecsRef` is held, no other methods may be
107    /// called on this `Buffer` since `IovecsRef` holds onto the mutable reference for this
108    /// `Buffer`.
109    ///
110    /// NOTE: The returned segments can only be accessed using user copy routines or Zircon system
111    /// calls (see the comment above for `peek_each_segment). The pointers returned are not _valid_
112    /// for any non-zero sized access (see Rust's std::ptr documentation).
113    fn peek_all_segments_as_iovecs(&mut self) -> Result<IovecsRef<'_, syncio::zxio::iovec>, Errno> {
114        IovecsRef::new(self)
115    }
116}
117
118/// Attempts to perform some I/O with the iovec segments of `Buffer`.
119///
120/// Returns `None` if the I/O can not be performed with iovecs (when unified
121/// aspaces is disabled or the `Buffer` does not support I/O on its segments
122/// directly).
123///
124/// NOTE: The segments can only be accessed using user copy routines or Zircon system calls (see the
125/// comment above for `peek_each_segment). The pointers returned are not _valid_ for any non-zero
126/// sized access (see Rust's std::ptr documentation).
127pub fn with_iovec_segments<B: Buffer + ?Sized, I: Iovec, T>(
128    data: &mut B,
129    f: impl FnOnce(&mut [I]) -> Result<T, Errno>,
130) -> Option<Result<T, Errno>> {
131    if !UNIFIED_ASPACES_ENABLED {
132        return None;
133    }
134
135    match IovecsRef::new(data) {
136        Ok(mut o) => Some(f(&mut o)),
137        Err(e) => {
138            if e.code == ENOTSUP {
139                None
140            } else {
141                Some(Err(e))
142            }
143        }
144    }
145}
146
147/// The OutputBuffer allows for writing bytes to a buffer.
148/// A single OutputBuffer will only write up to MAX_RW_COUNT bytes which is the maximum size of a
149/// single operation.
150pub trait OutputBuffer: Buffer {
151    /// Calls `callback` for each segment to write data for. `callback` must returns the number of
152    /// bytes actually written. When it returns less than the size of the input buffer, the write
153    /// is stopped.
154    ///
155    /// Returns the total number of bytes written.
156    fn write_each(&mut self, callback: &mut OutputBufferCallback<'_>) -> Result<usize, Errno>;
157
158    /// Returns the number of bytes available to be written into the buffer.
159    fn available(&self) -> usize;
160
161    /// Returns the number of bytes already written into the buffer.
162    fn bytes_written(&self) -> usize;
163
164    /// Fills this buffer with zeros.
165    fn zero(&mut self) -> Result<usize, Errno>;
166
167    /// Advance the output buffer by `length` bytes.
168    ///
169    /// # Safety
170    ///
171    /// The caller must guarantee that the length bytes are initialized.
172    unsafe fn advance(&mut self, length: usize) -> Result<(), Errno>;
173
174    /// Write the content of `buffer` into this buffer. If this buffer is too small, the write will
175    /// be partial.
176    ///
177    /// Returns the number of bytes written in this buffer.
178    fn write(&mut self, buffer: &[u8]) -> Result<usize, Errno> {
179        let mut buffer = slice_to_maybe_uninit(buffer);
180
181        self.write_each(&mut move |data| {
182            let size = std::cmp::min(buffer.len(), data.len());
183            let (to_clone, remaining) = buffer.split_at(size);
184            data[0..size].clone_from_slice(to_clone);
185            buffer = remaining;
186            Ok(size)
187        })
188    }
189
190    /// Write the content of `buffer` into this buffer. It is an error to pass a buffer larger than
191    /// the number of bytes available in this buffer. In that case, the content of the buffer after
192    /// the operation is unspecified.
193    ///
194    /// In case of success, always returns `buffer.len()`.
195    fn write_all(&mut self, buffer: &[u8]) -> Result<usize, Errno> {
196        let size = self.write(buffer)?;
197        if size != buffer.len() { error!(EINVAL) } else { Ok(size) }
198    }
199
200    /// Write the content of the given `InputBuffer` into this buffer. The number of bytes written
201    /// will be the smallest between the number of bytes available in this buffer and in the
202    /// `InputBuffer`.
203    ///
204    /// Returns the number of bytes read and written.
205    fn write_buffer(&mut self, input: &mut dyn InputBuffer) -> Result<usize, Errno> {
206        self.write_each(&mut move |data| {
207            let size = std::cmp::min(data.len(), input.available());
208            input.read_exact(&mut data[0..size])
209        })
210    }
211}
212
213/// The callback for `InputBuffer::peek_each` and `InputBuffer::read_each`. The callback is passed
214/// the buffers to write to in order, and must return for each, how many bytes has been read.
215
216pub type InputBufferCallback<'a> = dyn FnMut(&[u8]) -> Result<usize, Errno> + 'a;
217
218/// The InputBuffer allows for reading bytes from a buffer.
219/// A single InputBuffer will only read up to MAX_RW_COUNT bytes which is the maximum size of a
220/// single operation.
221pub trait InputBuffer: Buffer {
222    /// Calls `callback` for each segment to peek data from. `callback` must returns the number of
223    /// bytes actually peeked. When it returns less than the size of the output buffer, the read
224    /// is stopped.
225    ///
226    /// Returns the total number of bytes peeked.
227    fn peek_each(&mut self, callback: &mut InputBufferCallback<'_>) -> Result<usize, Errno>;
228
229    /// Returns the number of bytes available to be read from the buffer.
230    fn available(&self) -> usize;
231
232    /// Returns the number of bytes already read from the buffer.
233    fn bytes_read(&self) -> usize;
234
235    /// Clear the remaining content in the buffer. Returns the number of bytes swallowed. After this
236    /// method returns, `available()` will returns 0. This does not touch the data in the buffer.
237    fn drain(&mut self) -> usize;
238
239    /// Consumes `length` bytes of data from this buffer.
240    fn advance(&mut self, length: usize) -> Result<(), Errno>;
241
242    /// Calls `callback` for each segment to read data from. `callback` must returns the number of
243    /// bytes actually read. When it returns less than the size of the output buffer, the read
244    /// is stopped.
245    ///
246    /// Returns the total number of bytes read.
247    fn read_each(&mut self, callback: &mut InputBufferCallback<'_>) -> Result<usize, Errno> {
248        let length = self.peek_each(callback)?;
249        self.advance(length)?;
250        Ok(length)
251    }
252
253    /// Read all the remaining content in this buffer and returns it as a `Vec`.
254    fn read_all(&mut self) -> Result<Vec<u8>, Errno> {
255        let result = self.peek_all()?;
256        let drain_result = self.drain();
257        assert!(result.len() == drain_result);
258        Ok(result)
259    }
260
261    /// Peek all the remaining content in this buffer and returns it as a `Vec`.
262    fn peek_all(&mut self) -> Result<Vec<u8>, Errno> {
263        // SAFETY: self.peek returns the number of bytes read.
264        unsafe {
265            read_to_vec::<u8, _>(self.available(), |buf| self.peek(buf).map(NumberOfElementsRead))
266        }
267    }
268
269    /// Peeks the content of this buffer into `buffer`.
270    /// If `buffer` is too small, the read will be partial.
271    /// If `buffer` is too large, the remaining bytes will be left untouched.
272    ///
273    /// Returns the number of bytes read from this buffer.
274    fn peek(&mut self, buffer: &mut [MaybeUninit<u8>]) -> Result<usize, Errno> {
275        let mut index = 0;
276        self.peek_each(&mut move |data| {
277            let data = slice_to_maybe_uninit(data);
278            let size = std::cmp::min(buffer.len() - index, data.len());
279            buffer[index..index + size].clone_from_slice(&data[..size]);
280            index += size;
281            Ok(size)
282        })
283    }
284
285    /// Write the content of this buffer into `buffer`.
286    /// If `buffer` is too small, the read will be partial.
287    /// If `buffer` is too large, the remaining bytes will be left untouched.
288    ///
289    /// Returns the number of bytes read from this buffer.
290    fn read(&mut self, buffer: &mut [MaybeUninit<u8>]) -> Result<usize, Errno> {
291        let length = self.peek(buffer)?;
292        self.advance(length)?;
293        Ok(length)
294    }
295
296    /// Read the exact number of bytes required to fill buf.
297    ///
298    /// If `buffer` is larger than the number of available bytes, an error will be returned.
299    ///
300    /// In case of success, always returns `buffer.len()`.
301    fn read_exact(&mut self, buffer: &mut [MaybeUninit<u8>]) -> Result<usize, Errno> {
302        let size = self.read(buffer)?;
303        if size != buffer.len() { error!(EINVAL) } else { Ok(size) }
304    }
305}
306
307pub trait InputBufferExt: InputBuffer {
308    /// Reads exactly `len` bytes into a returned `Vec`.
309    ///
310    /// Returns an error if `len` is larger than the number of available bytes.
311    fn read_to_vec_exact(&mut self, len: usize) -> Result<Vec<u8>, Errno> {
312        // SAFETY: `data.read_exact` returns `len` bytes on success.
313        unsafe { read_to_vec::<u8, _>(len, |buf| self.read_exact(buf).map(NumberOfElementsRead)) }
314    }
315
316    /// Reads up to `limit` bytes into a returned `Vec`.
317    fn read_to_vec_limited(&mut self, limit: usize) -> Result<Vec<u8>, Errno> {
318        // SAFETY: `data.read` returns the number of bytes read.
319        unsafe { read_to_vec::<u8, _>(limit, |buf| self.read(buf).map(NumberOfElementsRead)) }
320    }
321
322    /// Reads bytes into the array.
323    ///
324    /// Returns an error if `N` is larger than the number of available bytes.
325    fn read_to_array<const N: usize>(&mut self) -> Result<[u8; N], Errno> {
326        // SAFETY: `data.read_exact` returns `N` bytes on success.
327        unsafe {
328            read_to_array::<_, _, N>(|buf| {
329                self.read_exact(buf).map(|bytes_read| debug_assert_eq!(bytes_read, buf.len()))
330            })
331        }
332    }
333
334    /// Interprets the buffer as an object.
335    ///
336    /// Returns an error if the buffer does not have enough bytes to represent the
337    /// object.
338    fn read_to_object<T: FromBytes>(&mut self) -> Result<T, Errno> {
339        // SAFETY: the callback returns successfully only if the required number of
340        // bytes were read.
341        unsafe {
342            read_to_object_as_bytes(|buf| {
343                if self.read(buf)? != buf.len() { error!(EINVAL) } else { Ok(()) }
344            })
345        }
346    }
347}
348
349impl InputBufferExt for dyn InputBuffer + '_ {}
350impl<T: InputBuffer> InputBufferExt for T {}
351
352/// An OutputBuffer that write data to user space memory through a `TaskMemoryAccessor`.
353pub struct UserBuffersOutputBuffer<'a, M> {
354    mm: &'a M,
355    buffers: UserBuffers,
356    available: usize,
357    bytes_written: usize,
358}
359
360impl<'a, M> std::fmt::Debug for UserBuffersOutputBuffer<'a, M> {
361    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
362        f.debug_struct("UserBuffersOutputBuffer")
363            .field("buffers", &self.buffers)
364            .field("available", &self.available)
365            .field("bytes_written", &self.bytes_written)
366            .finish()
367    }
368}
369
370impl<'a, M: TaskMemoryAccessor> UserBuffersOutputBuffer<'a, M> {
371    fn new_inner(mm: &'a M, mut buffers: UserBuffers) -> Result<Self, Errno> {
372        let available = UserBuffer::cap_buffers_to_max_rw_count(
373            mm.maximum_valid_address().ok_or_else(|| errno!(EINVAL))?,
374            &mut buffers,
375        )?;
376        // Reverse the buffers as the element will be removed as they are handled.
377        buffers.reverse();
378        Ok(Self { mm, buffers, available, bytes_written: 0 })
379    }
380
381    fn write_each_inner<B: AsRef<[u8]>, F: FnMut(usize) -> Result<B, Errno>>(
382        &mut self,
383        mut callback: F,
384    ) -> Result<usize, Errno> {
385        let mut bytes_written = 0;
386        while let Some(mut buffer) = self.buffers.pop() {
387            if buffer.is_null() {
388                continue;
389            }
390
391            let bytes = callback(buffer.length)?;
392            let bytes = bytes.as_ref();
393
394            bytes_written += self.mm.write_memory(buffer.address, bytes)?;
395            let bytes_len = bytes.len();
396            buffer.advance(bytes_len)?;
397            self.available -= bytes_len;
398            self.bytes_written += bytes_len;
399            if !buffer.is_empty() {
400                self.buffers.push(buffer);
401                break;
402            }
403        }
404        Ok(bytes_written)
405    }
406}
407
408impl<'a> UserBuffersOutputBuffer<'a, CurrentTask> {
409    pub fn unified_new(task: &'a CurrentTask, buffers: UserBuffers) -> Result<Self, Errno> {
410        Self::new_inner(task, buffers)
411    }
412
413    pub fn unified_new_at(
414        task: &'a CurrentTask,
415        address: UserAddress,
416        length: usize,
417    ) -> Result<Self, Errno> {
418        Self::unified_new(task, smallvec![UserBuffer { address, length }])
419    }
420}
421
422impl<'a> UserBuffersOutputBuffer<'a, Task> {
423    pub fn syscall_new(task: &'a Task, buffers: UserBuffers) -> Result<Self, Errno> {
424        Self::new_inner(task, buffers)
425    }
426}
427
428impl<'a, M: TaskMemoryAccessor> Buffer for UserBuffersOutputBuffer<'a, M> {
429    fn segments_count(&self) -> Result<usize, Errno> {
430        Ok(self.buffers.len())
431    }
432
433    fn peek_each_segment(
434        &mut self,
435        callback: &mut PeekBufferSegmentsCallback<'_>,
436    ) -> Result<(), Errno> {
437        // This `UserBuffersOutputBuffer` made sure that each segment only pointed
438        // to valid user-space address ranges on creation so each `buffer` is
439        // safe to write to.
440        for buffer in self.buffers.iter().rev() {
441            if buffer.is_null() {
442                continue;
443            }
444            callback(buffer)
445        }
446
447        Ok(())
448    }
449}
450
451impl<'a, M: TaskMemoryAccessor> OutputBuffer for UserBuffersOutputBuffer<'a, M> {
452    fn write(&mut self, mut bytes: &[u8]) -> Result<usize, Errno> {
453        self.write_each_inner(|buflen| {
454            let bytes_len = std::cmp::min(bytes.len(), buflen);
455            let (to_write, remaining) = bytes.split_at(bytes_len);
456            bytes = remaining;
457            Ok(to_write)
458        })
459    }
460
461    fn write_each(&mut self, callback: &mut OutputBufferCallback<'_>) -> Result<usize, Errno> {
462        self.write_each_inner(|buflen| {
463            // SAFETY: `callback` returns the number of bytes read on success.
464            unsafe {
465                read_to_vec::<u8, _>(buflen, |buf| {
466                    let result = callback(buf)?;
467                    if result > buflen {
468                        return error!(EINVAL);
469                    }
470                    Ok(NumberOfElementsRead(result))
471                })
472            }
473        })
474    }
475
476    fn available(&self) -> usize {
477        self.available
478    }
479
480    fn bytes_written(&self) -> usize {
481        self.bytes_written
482    }
483
484    fn zero(&mut self) -> Result<usize, Errno> {
485        let mut bytes_written = 0;
486        while let Some(mut buffer) = self.buffers.pop() {
487            if buffer.is_null() {
488                continue;
489            }
490
491            let count = self.mm.zero(buffer.address, buffer.length)?;
492            buffer.advance(count)?;
493            bytes_written += count;
494
495            self.available -= count;
496            self.bytes_written += count;
497
498            if !buffer.is_empty() {
499                self.buffers.push(buffer);
500                break;
501            }
502        }
503
504        Ok(bytes_written)
505    }
506
507    unsafe fn advance(&mut self, mut length: usize) -> Result<(), Errno> {
508        if length > self.available() {
509            return error!(EINVAL);
510        }
511
512        while let Some(mut buffer) = self.buffers.pop() {
513            if buffer.is_null() {
514                continue;
515            }
516
517            let advance_by = std::cmp::min(length, buffer.length);
518            buffer.advance(advance_by)?;
519            self.available -= advance_by;
520            self.bytes_written += advance_by;
521            if !buffer.is_empty() {
522                self.buffers.push(buffer);
523                break;
524            }
525            length -= advance_by;
526        }
527
528        Ok(())
529    }
530}
531
532/// An InputBuffer that read data from user space memory through a `TaskMemoryAccessor`.
533pub struct UserBuffersInputBuffer<'a, M> {
534    mm: &'a M,
535    buffers: UserBuffers,
536    available: usize,
537    bytes_read: usize,
538}
539
540impl<'a, M> std::fmt::Debug for UserBuffersInputBuffer<'a, M> {
541    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
542        f.debug_struct("UserBuffersInputBuffer")
543            .field("buffers", &self.buffers)
544            .field("available", &self.available)
545            .field("bytes_read", &self.bytes_read)
546            .finish()
547    }
548}
549
550impl<'a, M: TaskMemoryAccessor> UserBuffersInputBuffer<'a, M> {
551    fn new_inner(mm: &'a M, mut buffers: UserBuffers) -> Result<Self, Errno> {
552        let available = UserBuffer::cap_buffers_to_max_rw_count(
553            mm.maximum_valid_address().ok_or_else(|| errno!(EINVAL))?,
554            &mut buffers,
555        )?;
556        // Reverse the buffers as the element will be removed as they are handled.
557        buffers.reverse();
558        Ok(Self { mm, buffers, available, bytes_read: 0 })
559    }
560
561    fn peek_each_inner<F: FnMut(&UserBuffer, usize) -> Result<usize, Errno>>(
562        &mut self,
563        mut callback: F,
564    ) -> Result<usize, Errno> {
565        let mut read = 0;
566        for buffer in self.buffers.iter().rev() {
567            if buffer.is_null() {
568                continue;
569            }
570
571            let result = callback(buffer, read)?;
572            if result > buffer.length {
573                return error!(EINVAL);
574            }
575            read += result;
576            if result != buffer.length {
577                break;
578            }
579        }
580        Ok(read)
581    }
582}
583
584impl<'a> UserBuffersInputBuffer<'a, CurrentTask> {
585    pub fn unified_new(task: &'a CurrentTask, buffers: UserBuffers) -> Result<Self, Errno> {
586        Self::new_inner(task, buffers)
587    }
588
589    pub fn unified_new_at(
590        task: &'a CurrentTask,
591        address: UserAddress,
592        length: usize,
593    ) -> Result<Self, Errno> {
594        Self::unified_new(task, smallvec![UserBuffer { address, length }])
595    }
596}
597
598impl<'a> UserBuffersInputBuffer<'a, Task> {
599    pub fn syscall_new(task: &'a Task, buffers: UserBuffers) -> Result<Self, Errno> {
600        Self::new_inner(task, buffers)
601    }
602}
603
604impl<'a, M: TaskMemoryAccessor> Buffer for UserBuffersInputBuffer<'a, M> {
605    fn segments_count(&self) -> Result<usize, Errno> {
606        Ok(self.buffers.iter().filter(|b| b.is_null()).count())
607    }
608
609    fn peek_each_segment(
610        &mut self,
611        callback: &mut PeekBufferSegmentsCallback<'_>,
612    ) -> Result<(), Errno> {
613        // This `UserBuffersInputBuffer` made sure that each segment only pointed
614        // to valid user-space address ranges on creation so each `buffer` is
615        // safe to read from.
616        for buffer in self.buffers.iter().rev() {
617            if buffer.is_null() {
618                continue;
619            }
620            callback(buffer)
621        }
622
623        Ok(())
624    }
625}
626
627impl<'a, M: TaskMemoryAccessor> InputBuffer for UserBuffersInputBuffer<'a, M> {
628    fn peek(&mut self, uninit_bytes: &mut [MaybeUninit<u8>]) -> Result<usize, Errno> {
629        self.peek_each_inner(|buffer, read_so_far| {
630            let read_to = &mut uninit_bytes[read_so_far..];
631            let read_count = std::cmp::min(buffer.length, read_to.len());
632            let read_to = &mut read_to[..read_count];
633            let read_bytes = self.mm.read_memory(buffer.address, read_to)?;
634            debug_assert_eq!(read_bytes.len(), read_count);
635            Ok(read_count)
636        })
637    }
638
639    fn peek_each(&mut self, callback: &mut InputBufferCallback<'_>) -> Result<usize, Errno> {
640        self.peek_each_inner(|buffer, _read_so_far| {
641            let bytes = self.mm.read_memory_to_vec(buffer.address, buffer.length)?;
642            callback(&bytes)
643        })
644    }
645
646    fn drain(&mut self) -> usize {
647        let result = self.available;
648        self.bytes_read += self.available;
649        self.available = 0;
650        self.buffers.clear();
651        result
652    }
653
654    fn advance(&mut self, mut length: usize) -> Result<(), Errno> {
655        if length > self.available {
656            return error!(EINVAL);
657        }
658        self.available -= length;
659        self.bytes_read += length;
660        while let Some(mut buffer) = self.buffers.pop() {
661            if length < buffer.length {
662                buffer.advance(length)?;
663                self.buffers.push(buffer);
664                return Ok(());
665            }
666            length -= buffer.length;
667            if length == 0 {
668                return Ok(());
669            }
670        }
671        if length != 0 { error!(EINVAL) } else { Ok(()) }
672    }
673
674    fn available(&self) -> usize {
675        self.available
676    }
677    fn bytes_read(&self) -> usize {
678        self.bytes_read
679    }
680}
681
682/// An OutputBuffer that write data to an internal buffer.
683#[derive(Debug)]
684pub struct VecOutputBuffer {
685    buffer: Vec<u8>,
686    // Used to keep track of the requested capacity. `Vec::with_capacity` may
687    // allocate more than the requested capacity so we can't rely on
688    // `Vec::capacity` to return the expected capacity.
689    capacity: usize,
690}
691
692impl VecOutputBuffer {
693    pub fn new(capacity: usize) -> Self {
694        Self { buffer: Vec::with_capacity(capacity), capacity }
695    }
696
697    pub fn data(&self) -> &[u8] {
698        &self.buffer
699    }
700
701    pub fn reset(&mut self) {
702        self.buffer.clear()
703    }
704}
705
706impl From<VecOutputBuffer> for Vec<u8> {
707    fn from(data: VecOutputBuffer) -> Self {
708        data.buffer
709    }
710}
711
712impl Buffer for VecOutputBuffer {
713    fn segments_count(&self) -> Result<usize, Errno> {
714        Ok(1)
715    }
716
717    fn peek_each_segment(
718        &mut self,
719        callback: &mut PeekBufferSegmentsCallback<'_>,
720    ) -> Result<(), Errno> {
721        let current_len = self.buffer.len();
722        let buffer = &mut self.buffer.spare_capacity_mut()[..self.capacity - current_len];
723        callback(&UserBuffer {
724            address: UserAddress::from(buffer.as_mut_ptr() as u64),
725            length: buffer.len(),
726        });
727
728        Ok(())
729    }
730}
731
732impl OutputBuffer for VecOutputBuffer {
733    fn write_each(&mut self, callback: &mut OutputBufferCallback<'_>) -> Result<usize, Errno> {
734        let current_len = self.buffer.len();
735        let written =
736            callback(&mut self.buffer.spare_capacity_mut()[..self.capacity - current_len])?;
737        if current_len + written > self.capacity {
738            return error!(EINVAL);
739        }
740        // SAFETY: the vector is now initialized for an extra `written` bytes.
741        unsafe { self.buffer.set_len(current_len + written) }
742        Ok(written)
743    }
744
745    fn available(&self) -> usize {
746        self.capacity - self.buffer.len()
747    }
748
749    fn bytes_written(&self) -> usize {
750        self.buffer.len()
751    }
752
753    fn zero(&mut self) -> Result<usize, Errno> {
754        let zeroed = self.capacity - self.buffer.len();
755        self.buffer.resize(self.capacity, 0);
756        Ok(zeroed)
757    }
758
759    unsafe fn advance(&mut self, length: usize) -> Result<(), Errno> {
760        if length > self.available() {
761            return error!(EINVAL);
762        }
763
764        self.capacity -= length;
765        let current_len = self.buffer.len();
766        // SAFETY: We checked that length <= self.available(), and we updated self.capacity.
767        // self.available() is self.capacity - self.buffer.len().
768        // So length <= self.capacity - self.buffer.len()
769        // self.buffer.len() + length <= self.capacity.
770        // The buffer has at least self.capacity capacity (see VecOutputBuffer::new).
771        unsafe { self.buffer.set_len(current_len + length) };
772        Ok(())
773    }
774}
775
776/// An InputBuffer that read data from an internal buffer.
777#[derive(Debug)]
778pub struct VecInputBuffer {
779    buffer: Vec<u8>,
780
781    // Invariant: `bytes_read <= buffer.len()` at all times.
782    bytes_read: usize,
783}
784
785impl VecInputBuffer {
786    pub fn new(buffer: &[u8]) -> Self {
787        Self { buffer: buffer.to_vec(), bytes_read: 0 }
788    }
789}
790
791impl From<Vec<u8>> for VecInputBuffer {
792    fn from(buffer: Vec<u8>) -> Self {
793        Self { buffer, bytes_read: 0 }
794    }
795}
796
797impl Buffer for VecInputBuffer {
798    fn segments_count(&self) -> Result<usize, Errno> {
799        Ok(1)
800    }
801
802    fn peek_each_segment(
803        &mut self,
804        callback: &mut PeekBufferSegmentsCallback<'_>,
805    ) -> Result<(), Errno> {
806        let buffer = &self.buffer[self.bytes_read..];
807        callback(&UserBuffer {
808            address: UserAddress::from(buffer.as_ptr() as u64),
809            length: buffer.len(),
810        });
811
812        Ok(())
813    }
814}
815
816impl InputBuffer for VecInputBuffer {
817    fn peek_each(&mut self, callback: &mut InputBufferCallback<'_>) -> Result<usize, Errno> {
818        let read = callback(&self.buffer[self.bytes_read..])?;
819        if self.bytes_read + read > self.buffer.len() {
820            return error!(EINVAL);
821        }
822        debug_assert!(self.bytes_read <= self.buffer.len());
823        Ok(read)
824    }
825    fn advance(&mut self, length: usize) -> Result<(), Errno> {
826        if length > self.buffer.len() {
827            return error!(EINVAL);
828        }
829        self.bytes_read += length;
830        debug_assert!(self.bytes_read <= self.buffer.len());
831        Ok(())
832    }
833    fn available(&self) -> usize {
834        self.buffer.len() - self.bytes_read
835    }
836    fn bytes_read(&self) -> usize {
837        self.bytes_read
838    }
839    fn drain(&mut self) -> usize {
840        let result = self.available();
841        self.bytes_read += result;
842        result
843    }
844}
845
846impl VecInputBuffer {
847    /// Read an object from userspace memory and increment the read position.
848    ///
849    /// Returns an error if there is not enough available bytes compared to the size of `T`.
850    pub fn read_object<T: FromBytes>(&mut self) -> Result<T, Errno> {
851        let size = std::mem::size_of::<T>();
852        let end = self.bytes_read + size;
853        if end > self.buffer.len() {
854            return error!(EINVAL);
855        }
856        let obj =
857            T::read_from_bytes(&self.buffer[self.bytes_read..end]).map_err(|_| errno!(EINVAL))?;
858        self.bytes_read = end;
859        debug_assert!(self.bytes_read <= self.buffer.len());
860        Ok(obj)
861    }
862}
863
864#[cfg(test)]
865mod tests {
866    use super::*;
867    use crate::mm::{MemoryAccessor as _, PAGE_SIZE};
868    use crate::testing::*;
869    use usercopy::slice_to_maybe_uninit_mut;
870
871    #[test]
872    fn test_data_input_buffer() {
873        let mut executor = fuchsia_async::TestExecutor::new();
874        executor.run_singlethreaded(async {
875            spawn_kernel_and_run(async |locked, current_task| {
876                let page_size = *PAGE_SIZE;
877                let addr =
878                    map_memory(locked, &current_task, UserAddress::default(), 64 * page_size);
879
880                let data: Vec<u8> = (0..1024).map(|i| (i % 256) as u8).collect();
881                let mm = current_task.deref();
882                mm.write_memory(addr, &data).expect("failed to write test data");
883
884                let input_iovec = smallvec![
885                    UserBuffer { address: addr, length: 25 },
886                    UserBuffer {
887                        address: (addr + 64usize).expect("Memory mapped OOB!"),
888                        length: 12
889                    },
890                ];
891
892                // Test incorrect callback.
893                {
894                    let mut input_buffer =
895                        UserBuffersInputBuffer::unified_new(mm, input_iovec.clone())
896                            .expect("UserBuffersInputBuffer");
897                    assert!(input_buffer.peek_each(&mut |data| Ok(data.len() + 1)).is_err());
898                }
899
900                // Test drain
901                {
902                    let mut input_buffer =
903                        UserBuffersInputBuffer::unified_new(mm, input_iovec.clone())
904                            .expect("UserBuffersInputBuffer");
905                    assert_eq!(input_buffer.available(), 37);
906                    assert_eq!(input_buffer.bytes_read(), 0);
907                    assert_eq!(input_buffer.drain(), 37);
908                    assert_eq!(input_buffer.available(), 0);
909                    assert_eq!(input_buffer.bytes_read(), 37);
910                }
911
912                // Test read_all
913                {
914                    let mut input_buffer =
915                        UserBuffersInputBuffer::unified_new(mm, input_iovec.clone())
916                            .expect("UserBuffersInputBuffer");
917                    assert_eq!(input_buffer.available(), 37);
918                    assert_eq!(input_buffer.bytes_read(), 0);
919                    let buffer = input_buffer.read_all().expect("read_all");
920                    assert_eq!(input_buffer.available(), 0);
921                    assert_eq!(input_buffer.bytes_read(), 37);
922                    assert_eq!(buffer.len(), 37);
923                    assert_eq!(&data[..25], &buffer[..25]);
924                    assert_eq!(&data[64..76], &buffer[25..37]);
925                }
926
927                // Test read
928                {
929                    let mut input_buffer = UserBuffersInputBuffer::unified_new(mm, input_iovec)
930                        .expect("UserBuffersInputBuffer");
931                    let mut buffer = [0; 50];
932                    assert_eq!(input_buffer.available(), 37);
933                    assert_eq!(input_buffer.bytes_read(), 0);
934                    assert_eq!(
935                        input_buffer
936                            .read_exact(slice_to_maybe_uninit_mut(&mut buffer[0..20]))
937                            .expect("read"),
938                        20
939                    );
940                    assert_eq!(input_buffer.available(), 17);
941                    assert_eq!(input_buffer.bytes_read(), 20);
942                    assert_eq!(
943                        input_buffer
944                            .read_exact(slice_to_maybe_uninit_mut(&mut buffer[20..37]))
945                            .expect("read"),
946                        17
947                    );
948                    assert!(
949                        input_buffer
950                            .read_exact(slice_to_maybe_uninit_mut(&mut buffer[37..]))
951                            .is_err()
952                    );
953                    assert_eq!(input_buffer.available(), 0);
954                    assert_eq!(input_buffer.bytes_read(), 37);
955                    assert_eq!(&data[..25], &buffer[..25]);
956                    assert_eq!(&data[64..76], &buffer[25..37]);
957                }
958            })
959            .await;
960        });
961    }
962
963    #[test]
964    fn test_data_output_buffer() {
965        let mut executor = fuchsia_async::TestExecutor::new();
966        executor.run_singlethreaded(async {
967            spawn_kernel_and_run(async |locked, current_task| {
968                let page_size = *PAGE_SIZE;
969                let addr =
970                    map_memory(locked, &current_task, UserAddress::default(), 64 * page_size);
971
972                let output_iovec = smallvec![
973                    UserBuffer { address: addr, length: 25 },
974                    UserBuffer {
975                        address: (addr + 64usize).expect("Memory was mapped OOB!"),
976                        length: 12
977                    },
978                ];
979
980                let mm = current_task.deref();
981                let data: Vec<u8> = (0..1024).map(|i| (i % 256) as u8).collect();
982
983                // Test incorrect callback.
984                {
985                    let mut output_buffer =
986                        UserBuffersOutputBuffer::unified_new(mm, output_iovec.clone())
987                            .expect("UserBuffersOutputBuffer");
988                    assert!(output_buffer.write_each(&mut |data| Ok(data.len() + 1)).is_err());
989                }
990
991                // Test write
992                {
993                    let mut output_buffer = UserBuffersOutputBuffer::unified_new(mm, output_iovec)
994                        .expect("UserBuffersOutputBuffer");
995                    assert_eq!(output_buffer.available(), 37);
996                    assert_eq!(output_buffer.bytes_written(), 0);
997                    assert_eq!(output_buffer.write_all(&data[0..20]).expect("write"), 20);
998                    assert_eq!(output_buffer.available(), 17);
999                    assert_eq!(output_buffer.bytes_written(), 20);
1000                    assert_eq!(output_buffer.write_all(&data[20..37]).expect("write"), 17);
1001                    assert_eq!(output_buffer.available(), 0);
1002                    assert_eq!(output_buffer.bytes_written(), 37);
1003                    assert!(output_buffer.write_all(&data[37..50]).is_err());
1004
1005                    let buffer = current_task
1006                        .read_memory_to_array::<128>(addr)
1007                        .expect("failed to write test data");
1008                    assert_eq!(&data[0..25], &buffer[0..25]);
1009                    assert_eq!(&data[25..37], &buffer[64..76]);
1010                }
1011            })
1012            .await;
1013        });
1014    }
1015
1016    #[::fuchsia::test]
1017    fn test_vec_input_buffer() {
1018        let mut input_buffer = VecInputBuffer::new(b"helloworld");
1019        assert!(input_buffer.peek_each(&mut |data| Ok(data.len() + 1)).is_err());
1020
1021        let mut input_buffer = VecInputBuffer::new(b"helloworld");
1022        assert_eq!(input_buffer.bytes_read(), 0);
1023        assert_eq!(input_buffer.available(), 10);
1024        assert_eq!(input_buffer.drain(), 10);
1025        assert_eq!(input_buffer.bytes_read(), 10);
1026        assert_eq!(input_buffer.available(), 0);
1027
1028        let mut input_buffer = VecInputBuffer::new(b"helloworld");
1029        assert_eq!(input_buffer.bytes_read(), 0);
1030        assert_eq!(input_buffer.available(), 10);
1031        assert_eq!(&input_buffer.read_all().expect("read_all"), b"helloworld");
1032        assert_eq!(input_buffer.bytes_read(), 10);
1033        assert_eq!(input_buffer.available(), 0);
1034
1035        let mut input_buffer = VecInputBuffer::new(b"helloworld");
1036        let mut buffer = [0; 5];
1037        assert_eq!(
1038            input_buffer.read_exact(slice_to_maybe_uninit_mut(&mut buffer)).expect("read"),
1039            5
1040        );
1041        assert_eq!(input_buffer.bytes_read(), 5);
1042        assert_eq!(input_buffer.available(), 5);
1043        assert_eq!(&buffer, b"hello");
1044        assert_eq!(
1045            input_buffer.read_exact(slice_to_maybe_uninit_mut(&mut buffer)).expect("read"),
1046            5
1047        );
1048        assert_eq!(input_buffer.bytes_read(), 10);
1049        assert_eq!(input_buffer.available(), 0);
1050        assert_eq!(&buffer, b"world");
1051        assert!(input_buffer.read_exact(slice_to_maybe_uninit_mut(&mut buffer)).is_err());
1052
1053        // Test read_object
1054        let mut input_buffer = VecInputBuffer::new(b"hello");
1055        assert_eq!(input_buffer.bytes_read(), 0);
1056        let buffer: [u8; 3] = input_buffer.read_object().expect("read_object");
1057        assert_eq!(&buffer, b"hel");
1058        assert_eq!(input_buffer.bytes_read(), 3);
1059        let buffer: [u8; 2] = input_buffer.read_object().expect("read_object");
1060        assert_eq!(&buffer, b"lo");
1061        assert_eq!(input_buffer.bytes_read(), 5);
1062        assert!(input_buffer.read_object::<[u8; 1]>().is_err());
1063        assert_eq!(input_buffer.bytes_read(), 5);
1064
1065        let mut input_buffer = VecInputBuffer::new(b"hello");
1066        assert_eq!(input_buffer.bytes_read(), 0);
1067        assert!(input_buffer.read_object::<[u8; 100]>().is_err());
1068        assert_eq!(input_buffer.bytes_read(), 0);
1069    }
1070
1071    #[::fuchsia::test]
1072    fn test_vec_output_buffer() {
1073        let mut output_buffer = VecOutputBuffer::new(10);
1074        assert!(output_buffer.write_each(&mut |data| Ok(data.len() + 1)).is_err());
1075        assert_eq!(output_buffer.bytes_written(), 0);
1076        assert_eq!(output_buffer.available(), 10);
1077        assert_eq!(output_buffer.write_all(b"hello").expect("write"), 5);
1078        assert_eq!(output_buffer.bytes_written(), 5);
1079        assert_eq!(output_buffer.available(), 5);
1080        assert_eq!(output_buffer.data(), b"hello");
1081        assert_eq!(output_buffer.write_all(b"world").expect("write"), 5);
1082        assert_eq!(output_buffer.bytes_written(), 10);
1083        assert_eq!(output_buffer.available(), 0);
1084        assert_eq!(output_buffer.data(), b"helloworld");
1085        assert!(output_buffer.write_all(b"foo").is_err());
1086        let data: Vec<u8> = output_buffer.into();
1087        assert_eq!(data, b"helloworld".to_vec());
1088    }
1089
1090    #[::fuchsia::test]
1091    fn test_vec_write_buffer() {
1092        let mut input_buffer = VecInputBuffer::new(b"helloworld");
1093        let mut output_buffer = VecOutputBuffer::new(20);
1094        assert_eq!(output_buffer.write_buffer(&mut input_buffer).expect("write_buffer"), 10);
1095        assert_eq!(output_buffer.data(), b"helloworld");
1096    }
1097}