Initial reorg.
This is largely the same as #305, but updated for the current tree.
This commit is contained in:
133
crates/misc/py/src/memory.rs
Normal file
133
crates/misc/py/src/memory.rs
Normal file
@@ -0,0 +1,133 @@
|
||||
//! WebAssembly Memory API object.
|
||||
|
||||
extern crate alloc;
|
||||
|
||||
use pyo3::class::PyBufferProtocol;
|
||||
use pyo3::exceptions::BufferError;
|
||||
use pyo3::ffi;
|
||||
use pyo3::prelude::*;
|
||||
|
||||
use alloc::rc::Rc;
|
||||
use core::cell::RefCell;
|
||||
use core::ptr;
|
||||
use std::ffi::CStr;
|
||||
use std::os::raw::{c_int, c_void};
|
||||
|
||||
use wasmtime_environ::MemoryPlan;
|
||||
use wasmtime_jit::{Context, InstanceHandle};
|
||||
use wasmtime_runtime::{Export, VMMemoryDefinition, VMMemoryImport};
|
||||
|
||||
#[pyclass]
|
||||
pub struct Memory {
|
||||
pub context: Rc<RefCell<Context>>,
|
||||
pub instance: InstanceHandle,
|
||||
pub export_name: String,
|
||||
}
|
||||
|
||||
impl Memory {
|
||||
fn descriptor(&self) -> *mut VMMemoryDefinition {
|
||||
let mut instance = self.instance.clone();
|
||||
if let Some(Export::Memory { definition, .. }) = instance.lookup(&self.export_name) {
|
||||
definition
|
||||
} else {
|
||||
panic!("memory is expected");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Memory {
|
||||
pub fn get_plan(&self) -> MemoryPlan {
|
||||
let mut instance = self.instance.clone();
|
||||
if let Some(Export::Memory { memory, .. }) = instance.lookup(&self.export_name) {
|
||||
memory
|
||||
} else {
|
||||
panic!()
|
||||
}
|
||||
}
|
||||
|
||||
pub fn into_import(&self) -> VMMemoryImport {
|
||||
let mut instance = self.instance.clone();
|
||||
if let Some(Export::Memory {
|
||||
definition, vmctx, ..
|
||||
}) = instance.lookup(&self.export_name)
|
||||
{
|
||||
VMMemoryImport {
|
||||
from: definition,
|
||||
vmctx,
|
||||
}
|
||||
} else {
|
||||
panic!()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[pymethods]
|
||||
impl Memory {
|
||||
#[getter(current)]
|
||||
pub fn current(&self) -> u32 {
|
||||
let current_length = unsafe { (*self.descriptor()).current_length };
|
||||
(current_length >> 16) as u32
|
||||
}
|
||||
|
||||
pub fn grow(&self, _number: u32) -> u32 {
|
||||
(-1i32) as u32
|
||||
}
|
||||
}
|
||||
|
||||
#[pyproto]
|
||||
impl PyBufferProtocol for Memory {
|
||||
fn bf_getbuffer(&self, view: *mut ffi::Py_buffer, flags: c_int) -> PyResult<()> {
|
||||
if view.is_null() {
|
||||
return Err(BufferError::py_err("View is null"));
|
||||
}
|
||||
|
||||
unsafe {
|
||||
/*
|
||||
As a special case, for temporary buffers that are wrapped by
|
||||
PyMemoryView_FromBuffer() or PyBuffer_FillInfo() this field is NULL.
|
||||
In general, exporting objects MUST NOT use this scheme.
|
||||
*/
|
||||
(*view).obj = ptr::null_mut();
|
||||
}
|
||||
|
||||
let readonly = if (flags & ffi::PyBUF_WRITABLE) == ffi::PyBUF_WRITABLE {
|
||||
0
|
||||
} else {
|
||||
1
|
||||
};
|
||||
|
||||
let VMMemoryDefinition {
|
||||
base,
|
||||
current_length,
|
||||
} = unsafe { *self.descriptor() };
|
||||
|
||||
unsafe {
|
||||
(*view).buf = base as *mut c_void;
|
||||
(*view).len = current_length as isize;
|
||||
(*view).readonly = readonly;
|
||||
(*view).itemsize = 1;
|
||||
|
||||
(*view).format = ptr::null_mut();
|
||||
if (flags & ffi::PyBUF_FORMAT) == ffi::PyBUF_FORMAT {
|
||||
let msg = CStr::from_bytes_with_nul(b"B\0").unwrap();
|
||||
(*view).format = msg.as_ptr() as *mut _;
|
||||
}
|
||||
|
||||
(*view).ndim = 1;
|
||||
(*view).shape = ptr::null_mut();
|
||||
if (flags & ffi::PyBUF_ND) == ffi::PyBUF_ND {
|
||||
(*view).shape = (&((*view).len)) as *const _ as *mut _;
|
||||
}
|
||||
|
||||
(*view).strides = ptr::null_mut();
|
||||
if (flags & ffi::PyBUF_STRIDES) == ffi::PyBUF_STRIDES {
|
||||
(*view).strides = &((*view).itemsize) as *const _ as *mut _;
|
||||
}
|
||||
|
||||
(*view).suboffsets = ptr::null_mut();
|
||||
(*view).internal = ptr::null_mut();
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user