1use 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
19pub type OutputBufferCallback<'a> = dyn FnMut(&mut [MaybeUninit<u8>]) -> Result<usize, Errno> + 'a;
22
23fn slice_to_maybe_uninit(buffer: &[u8]) -> &[MaybeUninit<u8>] {
24 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
52pub 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 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
86pub trait Buffer: std::fmt::Debug {
90 fn segments_count(&self) -> Result<usize, Errno>;
93
94 fn peek_each_segment(
100 &mut self,
101 callback: &mut PeekBufferSegmentsCallback<'_>,
102 ) -> Result<(), Errno>;
103
104 fn peek_all_segments_as_iovecs(&mut self) -> Result<IovecsRef<'_, syncio::zxio::iovec>, Errno> {
114 IovecsRef::new(self)
115 }
116}
117
118pub 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
147pub trait OutputBuffer: Buffer {
151 fn write_each(&mut self, callback: &mut OutputBufferCallback<'_>) -> Result<usize, Errno>;
157
158 fn available(&self) -> usize;
160
161 fn bytes_written(&self) -> usize;
163
164 fn zero(&mut self) -> Result<usize, Errno>;
166
167 unsafe fn advance(&mut self, length: usize) -> Result<(), Errno>;
173
174 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 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 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
213pub type InputBufferCallback<'a> = dyn FnMut(&[u8]) -> Result<usize, Errno> + 'a;
217
218pub trait InputBuffer: Buffer {
222 fn peek_each(&mut self, callback: &mut InputBufferCallback<'_>) -> Result<usize, Errno>;
228
229 fn available(&self) -> usize;
231
232 fn bytes_read(&self) -> usize;
234
235 fn drain(&mut self) -> usize;
238
239 fn advance(&mut self, length: usize) -> Result<(), Errno>;
241
242 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 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 fn peek_all(&mut self) -> Result<Vec<u8>, Errno> {
263 unsafe {
265 read_to_vec::<u8, _>(self.available(), |buf| self.peek(buf).map(NumberOfElementsRead))
266 }
267 }
268
269 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 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 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 fn read_to_vec_exact(&mut self, len: usize) -> Result<Vec<u8>, Errno> {
312 unsafe { read_to_vec::<u8, _>(len, |buf| self.read_exact(buf).map(NumberOfElementsRead)) }
314 }
315
316 fn read_to_vec_limited(&mut self, limit: usize) -> Result<Vec<u8>, Errno> {
318 unsafe { read_to_vec::<u8, _>(limit, |buf| self.read(buf).map(NumberOfElementsRead)) }
320 }
321
322 fn read_to_array<const N: usize>(&mut self) -> Result<[u8; N], Errno> {
326 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 fn read_to_object<T: FromBytes>(&mut self) -> Result<T, Errno> {
339 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
352pub 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 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 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 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
532pub 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 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 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#[derive(Debug)]
684pub struct VecOutputBuffer {
685 buffer: Vec<u8>,
686 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 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 unsafe { self.buffer.set_len(current_len + length) };
772 Ok(())
773 }
774}
775
776#[derive(Debug)]
778pub struct VecInputBuffer {
779 buffer: Vec<u8>,
780
781 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 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, ¤t_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 {
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 {
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 {
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 {
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, ¤t_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 {
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 {
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 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}