wasmtime 44.0.0

High-level API to expose the Wasmtime runtime
Documentation
use super::cvt;
use crate::prelude::*;
use crate::runtime::vm::sys::{capi, vm::MemoryImageSource};
use crate::runtime::vm::{HostAlignedByteCount, SendSyncPtr};
use core::ops::Range;
use core::ptr::{self, NonNull};
#[cfg(feature = "std")]
use std::{fs::File, path::Path};

#[cfg(feature = "std")]
pub fn open_file_for_mmap(_path: &Path) -> Result<File> {
    crate::bail!("not supported on this platform");
}

#[derive(Debug)]
pub struct Mmap {
    memory: SendSyncPtr<[u8]>,
}

impl Mmap {
    pub fn new_empty() -> Mmap {
        Mmap {
            memory: crate::vm::sys::empty_mmap(),
        }
    }

    pub fn new(size: HostAlignedByteCount) -> Result<Self> {
        let mut ptr = ptr::null_mut();
        cvt(unsafe {
            capi::wasmtime_mmap_new(
                size.byte_count(),
                capi::PROT_READ | capi::PROT_WRITE,
                &mut ptr,
            )
        })?;
        let memory = ptr::slice_from_raw_parts_mut(ptr.cast(), size.byte_count());
        let memory = SendSyncPtr::new(NonNull::new(memory).unwrap());
        Ok(Mmap { memory })
    }

    pub fn reserve(size: HostAlignedByteCount) -> Result<Self> {
        let mut ptr = ptr::null_mut();
        cvt(unsafe { capi::wasmtime_mmap_new(size.byte_count(), 0, &mut ptr) })?;
        let memory = ptr::slice_from_raw_parts_mut(ptr.cast(), size.byte_count());
        let memory = SendSyncPtr::new(NonNull::new(memory).unwrap());
        Ok(Mmap { memory })
    }

    #[cfg(feature = "std")]
    pub fn from_file(_file: &File) -> Result<Self> {
        crate::bail!("not supported on this platform");
    }

    pub unsafe fn make_accessible(
        &self,
        start: HostAlignedByteCount,
        len: HostAlignedByteCount,
    ) -> Result<()> {
        let ptr = self.memory.as_ptr();
        unsafe {
            cvt(capi::wasmtime_mprotect(
                ptr.byte_add(start.byte_count()).cast(),
                len.byte_count(),
                capi::PROT_READ | capi::PROT_WRITE,
            ))?;
        }

        Ok(())
    }

    #[inline]
    pub fn as_send_sync_ptr(&self) -> SendSyncPtr<u8> {
        self.memory.cast()
    }

    #[inline]
    pub fn len(&self) -> usize {
        self.memory.as_ptr().len()
    }

    pub unsafe fn make_executable(
        &self,
        range: Range<usize>,
        enable_branch_protection: bool,
    ) -> Result<()> {
        unsafe {
            let base = self.memory.as_ptr().byte_add(range.start).cast();
            let len = range.end - range.start;

            // not mapped into the C API at this time.
            let _ = enable_branch_protection;

            cvt(capi::wasmtime_mprotect(
                base,
                len,
                capi::PROT_READ | capi::PROT_EXEC,
            ))?;
        }
        Ok(())
    }

    pub unsafe fn make_readonly(&self, range: Range<usize>) -> Result<()> {
        unsafe {
            let base = self.memory.as_ptr().byte_add(range.start).cast();
            let len = range.end - range.start;

            cvt(capi::wasmtime_mprotect(base, len, capi::PROT_READ))?;
        }
        Ok(())
    }

    pub unsafe fn make_readwrite(&self, range: Range<usize>) -> Result<()> {
        unsafe {
            let base = self.memory.as_ptr().byte_add(range.start).cast();
            let len = range.end - range.start;

            cvt(capi::wasmtime_mprotect(
                base,
                len,
                capi::PROT_READ | capi::PROT_WRITE,
            ))?;
        }
        Ok(())
    }

    pub unsafe fn map_image_at(
        &self,
        image_source: &MemoryImageSource,
        source_offset: u64,
        memory_offset: HostAlignedByteCount,
        memory_len: HostAlignedByteCount,
    ) -> Result<()> {
        assert_eq!(source_offset, 0);
        unsafe {
            let base = self
                .memory
                .as_ptr()
                .byte_add(memory_offset.byte_count())
                .cast();
            cvt(capi::wasmtime_memory_image_map_at(
                image_source.image_ptr().as_ptr(),
                base,
                memory_len.byte_count(),
            ))
        }
    }
}

impl Drop for Mmap {
    fn drop(&mut self) {
        unsafe {
            let ptr = self.memory.as_ptr().cast();
            let len = self.memory.as_ptr().len();
            if len == 0 {
                return;
            }
            cvt(capi::wasmtime_munmap(ptr, len)).unwrap();
        }
    }
}