use super::memory::{Memory, MemoryBuffer};
use crate::store::AsStoreRef;
use crate::MemoryAccessError;
use std::mem::MaybeUninit;
use wasmer_types::Pages;
#[cfg(feature = "js")]
use crate::js::externals::memory_view as memory_view_impl;
#[cfg(feature = "jsc")]
use crate::jsc::externals::memory_view as memory_view_impl;
#[cfg(feature = "sys")]
use crate::sys::externals::memory_view as memory_view_impl;
#[derive(Debug)]
pub struct MemoryView<'a>(pub(crate) memory_view_impl::MemoryView<'a>);
impl<'a> MemoryView<'a> {
pub(crate) fn new(memory: &Memory, store: &'a (impl AsStoreRef + ?Sized)) -> Self {
MemoryView(memory_view_impl::MemoryView::new(&memory.0, store))
}
#[doc(hidden)]
pub fn data_ptr(&self) -> *mut u8 {
self.0.data_ptr()
}
pub fn data_size(&self) -> u64 {
self.0.data_size()
}
#[doc(hidden)]
pub unsafe fn data_unchecked(&self) -> &[u8] {
self.0.data_unchecked()
}
#[allow(clippy::mut_from_ref)]
#[doc(hidden)]
pub unsafe fn data_unchecked_mut(&self) -> &mut [u8] {
self.0.data_unchecked_mut()
}
pub fn size(&self) -> Pages {
self.0.size()
}
#[inline]
pub(crate) fn buffer(&'a self) -> MemoryBuffer<'a> {
MemoryBuffer(self.0.buffer())
}
pub fn read(&self, offset: u64, buf: &mut [u8]) -> Result<(), MemoryAccessError> {
self.0.read(offset, buf)
}
pub fn read_u8(&self, offset: u64) -> Result<u8, MemoryAccessError> {
self.0.read_u8(offset)
}
pub fn read_uninit<'b>(
&self,
offset: u64,
buf: &'b mut [MaybeUninit<u8>],
) -> Result<&'b mut [u8], MemoryAccessError> {
self.0.read_uninit(offset, buf)
}
pub fn write(&self, offset: u64, data: &[u8]) -> Result<(), MemoryAccessError> {
self.0.write(offset, data)
}
pub fn write_u8(&self, offset: u64, val: u8) -> Result<(), MemoryAccessError> {
self.0.write_u8(offset, val)
}
pub fn copy_to_vec(&self) -> Result<Vec<u8>, MemoryAccessError> {
let mut new_memory = Vec::new();
let mut offset = 0;
let mut chunk = [0u8; 40960];
while offset < self.data_size() {
let remaining = self.data_size() - offset;
let sublen = remaining.min(chunk.len() as u64) as usize;
self.read(offset, &mut chunk[..sublen])?;
new_memory.extend_from_slice(&chunk[..sublen]);
offset += sublen as u64;
}
Ok(new_memory)
}
pub fn copy_to_memory(&self, amount: u64, new_memory: &Self) -> Result<(), MemoryAccessError> {
let mut offset = 0;
let mut chunk = [0u8; 40960];
while offset < amount {
let remaining = amount - offset;
let sublen = remaining.min(chunk.len() as u64) as usize;
self.read(offset, &mut chunk[..sublen])?;
new_memory.write(offset, &chunk[..sublen])?;
offset += sublen as u64;
}
Ok(())
}
}