use crate::arena::Arena;
use crate::atom_table::Atom;
use crate::forms::Number;
+use crate::machine::machine_errors::DomainErrorType;
use crate::parser::ast::{Fixnum, MightNotFitInFixnum};
use dashu::Integer;
use std::ptr::NonNull;
pub struct FunctionDefinition {
- pub name: String,
+ pub name: Atom,
pub return_value: Atom,
pub args: Vec<Atom>,
}
return_type_name: Atom,
args: &[Arg],
arena: &mut Arena,
- structs_table: &HashMap<String, StructImpl>,
+ structs_table: &HashMap<Atom, StructImpl>,
) -> Result<Value, FfiError> {
let struct_type = structs_table
- .get(&*return_type_name.as_str())
+ .get(&return_type_name)
.ok_or(FfiError::StructNotFound(return_type_name))?;
let ffi_type = unsafe { *struct_type.ffi_type.as_raw_ptr() };
)
};
- let struct_val = struct_type.read(
- alloc.ptr.as_ptr(),
- &return_type_name.as_str(),
- structs_table,
- arena,
- );
+ let struct_val =
+ struct_type.read(alloc.ptr.as_ptr(), return_type_name, structs_table, arena);
drop(alloc);
&self,
args: &[Arg],
arena: &mut Arena,
- structs_table: &HashMap<String, StructImpl>,
+ structs_table: &HashMap<Atom, StructImpl>,
) -> Result<Value, FfiError> {
let call_fn: unsafe fn(&Self, &[Arg], &mut Arena) -> Result<Value, FfiError> =
match self.return_type {
#[derive(Debug, Default)]
pub struct ForeignFunctionTable {
- table: HashMap<String, FunctionImpl>,
- structs: HashMap<String, StructImpl>,
+ table: HashMap<Atom, FunctionImpl>,
+ structs: HashMap<Atom, StructImpl>,
}
#[derive(Clone, Debug)]
fn build(
&self,
- structs_table: &HashMap<String, StructImpl>,
+ structs_table: &HashMap<Atom, StructImpl>,
struct_args: &mut [Value],
) -> Result<FfiStruct, FfiError> {
let args = ArgValue::build_args(struct_args, &self.fields, structs_table)?;
fn read(
&self,
ptr: *mut c_void,
- struct_name: &str,
- struct_table: &HashMap<String, StructImpl>,
+ struct_name: Atom,
+ struct_table: &HashMap<Atom, StructImpl>,
arena: &mut Arena,
) -> Result<Value, FfiError> {
unsafe {
FfiType::F32 => read_float::<f32>(ptr, &mut layout),
FfiType::F64 => read_float::<f64>(ptr, &mut layout),
FfiType::Struct(substruct) => {
- let Some(substruct_type) = struct_table.get(&*substruct.as_str()) else {
+ let Some(substruct_type) = struct_table.get(substruct) else {
return Err(FfiError::StructNotFound(*substruct));
};
.map_err(|_| FfiError::LayoutError)?;
layout = new_layout;
let field_ptr = ptr.byte_offset(offset as isize);
- let struct_val = substruct_type.read(
- field_ptr,
- &substruct.as_str(),
- struct_table,
- arena,
- )?;
+ let struct_val =
+ substruct_type.read(field_ptr, *substruct, struct_table, arena)?;
Ok(struct_val)
}
- FfiType::Void => unreachable!("void is not a valid field type"),
+ FfiType::Void => return Err(FfiError::VoidArgumentType),
};
returns.push(val?);
}
- Ok(Value::Struct(struct_name.to_string(), returns))
+ Ok(Value::Struct(struct_name, returns))
}
}
}
}
}
- fn to_type(self, structs_table: &HashMap<String, StructImpl>) -> Result<Type, FfiError> {
+ fn to_type(self, structs_table: &HashMap<Atom, StructImpl>) -> Result<Type, FfiError> {
Ok(match self {
Self::I64 => libffi::middle::Type::i64(),
Self::I32 => libffi::middle::Type::i32(),
Self::F32 => libffi::middle::Type::f32(),
Self::F64 => libffi::middle::Type::f64(),
Self::Struct(struct_name) => structs_table
- .get(&*struct_name.as_str())
+ .get(&struct_name)
.ok_or(FfiError::StructNotFound(struct_name))?
.ffi_type
.clone(),
fn new(
val: &'val mut Value,
arg_type: &FfiType,
- structs_table: &HashMap<String, StructImpl>,
+ structs_table: &HashMap<Atom, StructImpl>,
) -> Result<Self, FfiError> {
match arg_type {
FfiType::U8 => Ok(Self::U8(val.as_int()?)),
FfiType::F64 => Ok(Self::F64(val.as_float()?)),
FfiType::Ptr => Ok(Self::Ptr(val.as_ptr()?, PhantomData)),
FfiType::CStr => Ok(Self::Ptr(val.as_ptr()?, PhantomData)),
- FfiType::Struct(atom) => {
- let (name, args) = val.as_struct()?;
+ FfiType::Struct(arg_type_name) => {
+ let (val_type_name, args) = val.as_struct()?;
- if &*atom.as_str() != name {
- return Err(FfiError::ValueCast);
+ if *arg_type_name != val_type_name {
+ return Err(FfiError::ValueCast(*arg_type_name, val_type_name));
}
- let Some(struct_type) = structs_table.get(name) else {
- return Err(FfiError::StructNotFound(*atom));
+ let Some(struct_type) = structs_table.get(&val_type_name) else {
+ return Err(FfiError::StructNotFound(*arg_type_name));
};
Ok(Self::Struct(struct_type.build(structs_table, args)?))
}
- FfiType::Void => Err(FfiError::InvalidArgumentType),
+ FfiType::Void => Err(FfiError::VoidArgumentType),
}
}
fn build_args(
args: &'val mut [Value],
types: &[FfiType],
- structs_table: &HashMap<String, StructImpl>,
+ structs_table: &HashMap<Atom, StructImpl>,
) -> Result<Vec<Self>, FfiError> {
if types.len() != args.len() {
return Err(FfiError::ArgCountMismatch);
self.table.extend(other.table);
}
- pub fn define_struct(&mut self, name: &str, atom_fields: Vec<Atom>) -> Result<(), FfiError> {
+ pub fn define_struct(&mut self, name: Atom, atom_fields: Vec<Atom>) -> Result<(), FfiError> {
let fields: Vec<_> = atom_fields.iter().map(FfiType::from_atom).collect();
let struct_type = libffi::middle::Type::structure(
fields
};
self.structs.insert(
- name.to_string(),
+ name,
StructImpl {
ffi_type: struct_type,
fields,
let mut ff_table: ForeignFunctionTable = Default::default();
let library = unsafe { Library::new(library_name) }?;
for function in functions {
- let symbol_name: CString = CString::new(function.name.clone())?;
+ let symbol_name: CString = CString::new(&*function.name.as_str())?;
let code_ptr: Symbol<*mut c_void> =
unsafe { library.get(symbol_name.as_bytes_with_nul()) }?;
let args: Vec<_> = function.args.iter().map(FfiType::from_atom).collect();
);
ff_table.table.insert(
- function.name.clone(),
+ function.name,
FunctionImpl {
cif,
args,
pub fn exec(
&mut self,
- fn_name: &str,
+ fn_name: Atom,
mut args: Vec<Value>,
arena: &mut Arena,
) -> Result<Value, FfiError> {
- let fn_impl = self.table.get(fn_name).ok_or(FfiError::FunctionNotFound)?;
+ let fn_impl = self
+ .table
+ .get(&fn_name)
+ .ok_or(FfiError::FunctionNotFound(fn_name))?;
let args = ArgValue::build_args(&mut args, &fn_impl.args, &self.structs)?;
}
match FfiType::from_atom(&kind) {
- FfiType::Void => Err(FfiError::InvalidFfiType),
+ FfiType::Void => Err(FfiError::VoidArgumentType),
FfiType::Bool => {
let val = args.as_int::<u8>()?;
let init = match val {
0 => false,
1 => true,
- _ => return Err(FfiError::ValueOutOfRange),
+ _ => return Err(FfiError::ValueOutOfRange(DomainErrorType::ZeroOrOne, args)),
};
allocate_primitive::<bool>(allocator, init, arena)
}
FfiType::F32 => allocate_primitive::<f32>(allocator, args.as_float()? as f32, arena),
FfiType::F64 => allocate_primitive::<f64>(allocator, args.as_float()?, arena),
FfiType::Ptr => allocate_primitive::<*mut c_void>(allocator, args.as_ptr()?, arena),
- FfiType::CStr => Err(FfiError::InvalidFfiType),
+ FfiType::CStr => Err(FfiError::CStrFieldType),
FfiType::Struct(_) => {
- let Some(struct_impl) = self.structs.get(&*kind.as_str()) else {
- return Err(FfiError::InvalidStruct);
+ let Some(struct_impl) = self.structs.get(&kind) else {
+ return Err(FfiError::StructNotFound(kind));
};
let (_, args) = args.as_struct()?;
let ptr = ptr.as_ptr()?;
let Some(ptr) = NonNull::new(ptr) else {
- return Err(FfiError::ValueOutOfRange);
+ return Err(FfiError::NullPtr);
};
match FfiType::from_atom(&kind) {
- FfiType::Void => Err(FfiError::InvalidFfiType),
+ FfiType::Void => Err(FfiError::VoidArgumentType),
FfiType::Bool | FfiType::U8 => Ok(unsafe { read_int::<u8>(ptr, arena) }),
FfiType::I8 => Ok(unsafe { read_int::<i8>(ptr, arena) }),
FfiType::U16 => Ok(unsafe { read_int::<u16>(ptr, arena) }),
unsafe { CStr::from_ptr(ptr.as_ptr().cast()) }.to_owned(),
)),
FfiType::Struct(_) => {
- let Some(struct_impl) = self.structs.get(&*kind.as_str()) else {
- return Err(FfiError::InvalidStruct);
+ let Some(struct_impl) = self.structs.get(&kind) else {
+ return Err(FfiError::StructNotFound(kind));
};
- struct_impl.read(ptr.as_ptr(), &kind.as_str(), &self.structs, arena)
+ struct_impl.read(ptr.as_ptr(), kind, &self.structs, arena)
}
}
}
let ptr = ptr.as_ptr()?;
let Some(ptr) = NonNull::new(ptr) else {
- return Err(FfiError::ValueOutOfRange);
+ return Err(FfiError::NullPtr);
};
match FfiType::from_atom(&kind) {
- FfiType::Void => return Err(FfiError::InvalidFfiType),
+ FfiType::Void => return Err(FfiError::VoidArgumentType),
FfiType::Bool => deallocate_primitive::<bool>(allocator, ptr),
FfiType::U8 => deallocate_primitive::<u8>(allocator, ptr),
FfiType::I8 => deallocate_primitive::<i8>(allocator, ptr),
FfiType::F32 => deallocate_primitive::<f32>(allocator, ptr),
FfiType::F64 => deallocate_primitive::<f64>(allocator, ptr),
FfiType::Ptr => deallocate_primitive::<*mut c_void>(allocator, ptr),
- FfiType::CStr => return Err(FfiError::InvalidFfiType),
+ FfiType::CStr => return Err(FfiError::CStrFieldType),
FfiType::Struct(_) => {
- let Some(struct_impl) = self.structs.get(&*kind.as_str()) else {
- return Err(FfiError::InvalidStruct);
+ let Some(struct_impl) = self.structs.get(&kind) else {
+ return Err(FfiError::StructNotFound(kind));
};
let layout = struct_impl.layout()?;
pub enum Value {
Number(Number),
CString(CString),
- Struct(String, Vec<Value>),
+ Struct(Atom, Vec<Value>),
}
impl Value {
match self {
Value::Number(Number::Integer(ibig_ptr)) => {
let ibig: &Integer = ibig_ptr;
- ibig.clone()
- .try_into()
- .map_err(|_| FfiError::ValueOutOfRange)
+ ibig.clone().try_into().map_err(|_| {
+ FfiError::ValueOutOfRange(DomainErrorType::FixedSizedInt, self.clone())
+ })
}
- Value::Number(Number::Fixnum(fixnum)) => fixnum
- .get_num()
- .try_into()
- .map_err(|_| FfiError::ValueOutOfRange),
- _ => Err(FfiError::ValueCast),
+ Value::Number(Number::Fixnum(fixnum)) => fixnum.get_num().try_into().map_err(|_| {
+ FfiError::ValueOutOfRange(DomainErrorType::FixedSizedInt, self.clone())
+ }),
+ _ => Err(FfiError::ValueOutOfRange(
+ DomainErrorType::FixedSizedInt,
+ self.clone(),
+ )),
}
}
fn as_float(&self) -> Result<f64, FfiError> {
match self {
&Value::Number(Number::Float(OrderedFloat(f))) => Ok(f),
- _ => Err(FfiError::ValueCast),
+ _ => Err(FfiError::ValueOutOfRange(
+ DomainErrorType::F64,
+ self.clone(),
+ )),
}
}
Value::Number(Number::Fixnum(fixnum)) => Ok(std::ptr::with_exposed_provenance_mut(
fixnum.get_num() as usize,
)),
- _ => Err(FfiError::ValueCast),
+ _ => Err(FfiError::ValueOutOfRange(
+ DomainErrorType::PtrLike,
+ self.clone(),
+ )),
}
}
- fn as_struct(&mut self) -> Result<(&str, &mut [Self]), FfiError> {
+ fn as_struct(&mut self) -> Result<(Atom, &mut [Self]), FfiError> {
match self {
- Value::Struct(name, values) => Ok((name, values)),
- _ => Err(FfiError::ValueCast),
+ Value::Struct(name, values) => Ok((*name, values)),
+ _ => Err(FfiError::ValueOutOfRange(
+ DomainErrorType::FfiStruct,
+ self.clone(),
+ )),
}
}
}
#[derive(Debug)]
pub enum FfiError {
- ValueCast,
- ValueOutOfRange,
- InvalidArgumentType,
- InvalidArgument,
- InvalidFfiType,
- InvalidStruct,
- FunctionNotFound,
+ ValueCast(Atom, Atom),
+ ValueOutOfRange(DomainErrorType, Value),
+ VoidArgumentType,
+ FunctionNotFound(Atom),
StructNotFound(Atom),
ArgCountMismatch,
AllocationFailed,
// LayoutError should never occour
LayoutError,
+ UnsupportedTypedef,
UnsupportedAbi,
+ CStrFieldType,
+ NullPtr,
}
impl std::fmt::Display for FfiError {
impl From<libffi::low::Error> for FfiError {
fn from(value: libffi::low::Error) -> Self {
match value {
- libffi::low::Error::Typedef => FfiError::InvalidFfiType,
+ libffi::low::Error::Typedef => FfiError::UnsupportedTypedef,
libffi::low::Error::Abi => FfiError::UnsupportedAbi,
}
}
use crate::parser::ast::*;
#[cfg(feature = "ffi")]
-use crate::ffi::FfiError;
+use crate::ffi::{self, FfiError};
use crate::forms::*;
use crate::functor_macro::*;
use crate::machine::heap::*;
}
}
+#[cfg(feature = "ffi")]
+impl DomainError for ffi::Value {
+ fn domain_error(self, machine_st: &mut MachineState, error: DomainErrorType) -> MachineError {
+ use ffi::Value;
+
+ match self {
+ Value::Number(number) => number.domain_error(machine_st, error),
+ Value::CString(cstring) => {
+ let str = cstring.to_string_lossy().into_owned();
+ let stub = functor!(
+ atom!("domain_error"),
+ [atom_as_cell((error.as_atom())), string(str)]
+ );
+
+ MachineError {
+ stub,
+ location: None,
+ }
+ }
+ Value::Struct(atom, _values) => atom_as_cell!(atom).domain_error(machine_st, error),
+ }
+ }
+}
+
#[inline(always)]
pub(super) fn functor_stub(name: Atom, arity: usize) -> MachineStub {
functor!(atom!("/"), [atom_as_cell(name), fixnum(arity)])
[atom_as_cell((atom!("process"))), cell(culprit)]
);
+ MachineError {
+ stub,
+ location: None,
+ }
+ }
+ ExistenceError::FfiFunction(atom) => {
+ let stub = functor!(
+ atom!("existence_error"),
+ [atom_as_cell((atom!("ffi_function"))), atom_as_cell(atom)]
+ );
+
+ MachineError {
+ stub,
+ location: None,
+ }
+ }
+ ExistenceError::FfiStructType(atom) => {
+ let stub = functor!(
+ atom!("existence_error"),
+ [atom_as_cell((atom!("ffi_struct_type"))), atom_as_cell(atom)]
+ );
+
MachineError {
stub,
location: None,
}
#[cfg(feature = "ffi")]
- pub(super) fn ffi_error(&self, err: FfiError, culprit: HeapCellValue) -> MachineError {
- let error_atom = match err {
- FfiError::ValueCast => atom!("value_cast"),
- FfiError::ValueOutOfRange => atom!("value_out_of_range"),
- FfiError::InvalidFfiType => atom!("invalid_ffi_type"),
- FfiError::InvalidArgumentType => atom!("invalid_argument_type"),
- FfiError::InvalidArgument => atom!("invalid_argument"),
- FfiError::InvalidStruct => atom!("invalid_struct"),
- FfiError::FunctionNotFound => atom!("function_not_found"),
- FfiError::StructNotFound(culprit) => {
+ pub(super) fn ffi_error(&mut self, err: FfiError) -> MachineError {
+ match err {
+ FfiError::ValueCast(expected, actual) => {
let stub = functor!(
- atom!("ffi_error"),
- [
- atom_as_cell((atom!("struct_not_found"))),
- atom_as_cell(culprit)
- ]
+ atom!("domain_error"),
+ [atom_as_cell(expected), atom_as_cell(actual)]
);
- return MachineError {
+ MachineError {
stub,
location: None,
- };
+ }
}
- FfiError::ArgCountMismatch => atom!("mismatched_argument_count"),
- FfiError::AllocationFailed => atom!("allocation_failed"),
- FfiError::LayoutError => atom!("layout_error"),
- FfiError::UnsupportedAbi => atom!("unsupported_abi"),
- };
- let stub = functor!(
- atom!("ffi_error"),
- [atom_as_cell(error_atom), cell(culprit)]
- );
-
- MachineError {
- stub,
- location: None,
+ FfiError::ValueOutOfRange(domain, culprit) => self.domain_error(domain, culprit),
+ FfiError::FunctionNotFound(name) => {
+ self.existence_error(ExistenceError::FfiFunction(name))
+ }
+ FfiError::StructNotFound(name) => {
+ self.existence_error(ExistenceError::FfiStructType(name))
+ }
+ FfiError::ArgCountMismatch => self.unreachable_error(),
+ FfiError::AllocationFailed => MachineError {
+ stub: functor!(atom!("resource_error"), [atom_as_cell((atom!("heap")))]),
+ location: None,
+ },
+ FfiError::LayoutError => self.representation_error(RepFlag::FfiLayout),
+ FfiError::UnsupportedTypedef => self.representation_error(RepFlag::FfiLayout),
+ FfiError::UnsupportedAbi => self.representation_error(RepFlag::FfiAbi),
+ FfiError::VoidArgumentType => self.domain_error(
+ DomainErrorType::FfiArgumentType,
+ atom_as_cell!(atom!("void")),
+ ),
+ FfiError::CStrFieldType => todo!(),
+ FfiError::NullPtr => self.domain_error(
+ DomainErrorType::NonNullPtr,
+ fixnum_as_cell!(Fixnum::build_with(0)),
+ ),
}
}
OperatorPriority,
Directive,
Allocator,
+ FfiStruct,
+ ZeroOrOne,
+ NonNullPtr,
+ PtrLike,
+ F64,
+ FfiArgument,
+ FfiArgumentType,
+ FixedSizedInt,
}
impl DomainErrorType {
DomainErrorType::OperatorPriority => atom!("operator_priority"),
DomainErrorType::Directive => atom!("directive"),
DomainErrorType::Allocator => atom!("allocator"),
+ DomainErrorType::ZeroOrOne => atom!("zero_or_one"),
+ DomainErrorType::FfiStruct => atom!("ffi_struct"),
+ DomainErrorType::NonNullPtr => atom!("non_null_pointer"),
+ DomainErrorType::PtrLike => atom!("pointer_like"),
+ DomainErrorType::F64 => atom!("f64"),
+ DomainErrorType::FfiArgument => atom!("ffi_argument"),
+ DomainErrorType::FfiArgumentType => atom!("ffi_argument_type"),
+ DomainErrorType::FixedSizedInt => atom!("fixed_sized_int"),
}
}
}
// MaxInteger,
// MinInteger,
Term,
+ FfiLayout,
+ FfiAbi,
}
impl RepFlag {
RepFlag::MaxArity => atom!("max_arity"),
RepFlag::Term => atom!("term"),
// RepFlag::MaxInteger => atom!("max_integer"),
- // RepFlag::MinInteger => atom!("min_integer")
+ // RepFlag::MinInteger => atom!("min_integer"),
+ RepFlag::FfiLayout => atom!("ffi_layout"),
+ RepFlag::FfiAbi => atom!("ffi_abi"),
}
}
}
SourceSink(HeapCellValue),
Stream(HeapCellValue),
Process(HeapCellValue),
+ FfiFunction(Atom),
+ FfiStructType(Atom),
}
#[derive(Debug)]
};
let return_value = cell_as_atom_cell!(self.machine_st.heap[s + 2]);
functions.push(FunctionDefinition {
- name: name.as_str().to_string(),
+ name,
args,
return_value: return_value.get_name(),
});
}
_ => {
- unreachable!()
- }
+ let err = self.machine_st.unreachable_error();
+ return Err(self.machine_st.error_form(err, stub_gen()))
+ }
)
}
if self
.foreign_function_table
.load_library(&library_name.as_str(), &functions)
- .is_ok()
+ .is_err()
{
- return Ok(());
+ self.machine_st.fail = true;
}
+
+ Ok(())
}
- Err(e) => return Err(e),
- };
+ Err(e) => Err(e),
+ }
+ } else {
+ let err = self
+ .machine_st
+ .type_error(ValidType::InCharacter, library_name);
+ Err(self.machine_st.error_form(err, stub_gen()))
}
- self.machine_st.fail = true;
- Ok(())
}
#[cfg(not(feature = "ffi"))]
if let Some(head) = iter.next() {
let head = self.machine_st.store(self.machine_st.deref(head));
- if let Some(struct_name) = self.machine_st.value_to_str_like(head) {
+ if let Some(struct_name) = head.to_atom() {
Ok(Value::Struct(
- struct_name.as_str().to_string(),
+ struct_name,
iter.map(|x| self.map_ffi_arg(x, stub_gen))
.collect::<Result<_, _>>()?,
))
Err(self.machine_st.error_form(err, src))
} else {
// first element of a struct needs to be the type
- Err(self.machine_st.error_form(
- self.machine_st.ffi_error(FfiError::ValueOutOfRange, head),
- stub_gen(),
- ))
+ let err = self.machine_st.type_error(ValidType::Atom, head);
+ Err(self.machine_st.error_form(err, stub_gen()))
}
} else {
// empty list is an invalid struct repr
- Err(self.machine_st.error_form(
- self.machine_st.ffi_error(FfiError::ValueOutOfRange, source),
- stub_gen(),
- ))
+ let err = self
+ .machine_st
+ .domain_error(DomainErrorType::FfiStruct, source);
+ Err(self.machine_st.error_form(err, stub_gen()))
}
} else if self.machine_st.deref(source).is_var() {
let err = self.machine_st.instantiation_error();
Err(self.machine_st.error_form(err, src))
} else {
- Err(self.machine_st.error_form(
- self.machine_st.ffi_error(FfiError::InvalidArgument, source),
- stub_gen(),
- ))
+ let err = self
+ .machine_st
+ .domain_error(DomainErrorType::FfiArgument, source);
+ Err(self.machine_st.error_form(err, stub_gen()))
}
}
#[cfg(feature = "ffi")]
{
- let function_name_arg = self.deref_register(1);
+ let function_name_arg = self.machine_st.store(self.deref_register(1));
let args_reg = self.deref_register(2);
let return_value = self.deref_register(3);
- if let Some(function_name) = self.machine_st.value_to_str_like(function_name_arg) {
+ if let Some(function_name) = function_name_arg.to_atom() {
match self.machine_st.try_from_list(args_reg, stub_gen) {
Ok(args) => {
let args = args
.collect::<Result<Vec<_>, _>>()?;
match self.foreign_function_table.exec(
- &function_name.as_str(),
+ function_name,
args,
&mut self.machine_st.arena,
) {
- Ok(result) => {
- return self.unify_ffi_result(return_value, result);
- }
+ Ok(result) => self.unify_ffi_result(return_value, result),
Err(e) => {
- let err = self.machine_st.ffi_error(e, function_name_arg);
- return Err(self.machine_st.error_form(err, stub_gen()));
+ let err = self.machine_st.ffi_error(e);
+ Err(self.machine_st.error_form(err, stub_gen()))
}
}
}
- Err(e) => return Err(e),
+ Err(e) => Err(e),
}
+ } else {
+ let err = self
+ .machine_st
+ .type_error(ValidType::Atom, function_name_arg);
+ Err(self.machine_st.error_form(err, stub_gen()))
}
-
- self.machine_st.fail = true;
- Ok(())
}
#[cfg(not(feature = "ffi"))]
},
Value::Struct(name, args) => {
let struct_value =
- resource_error_call_result!(self.machine_st, self.build_struct(&name, args));
+ resource_error_call_result!(self.machine_st, self.build_struct(name, args));
unify!(self.machine_st, return_value, struct_value);
}
}
#[cfg(feature = "ffi")]
- fn build_struct(&mut self, name: &str, mut args: Vec<Value>) -> Result<HeapCellValue, usize> {
- args.insert(0, Value::CString(CString::new(name).unwrap()));
+ fn build_struct(&mut self, name: Atom, mut args: Vec<Value>) -> Result<HeapCellValue, usize> {
+ args.insert(0, Value::CString(CString::new(&*name.as_str()).unwrap()));
let cells: Vec<_> = args
.into_iter()
&self.machine_st.atom_tbl,
&cstr.into_string().unwrap()
)),
- Value::Struct(name, struct_args) => self.build_struct(&name, struct_args)?,
+ Value::Struct(name, struct_args) => self.build_struct(name, struct_args)?,
})
})
.collect::<Result<_, usize>>()?;
#[cfg(feature = "ffi")]
{
- let struct_name_arg = self.deref_register(1);
+ let struct_name_arg = self.machine_st.store(self.deref_register(1));
let fields_reg = self.deref_register(2);
- if let Some(struct_name) = self.machine_st.value_to_str_like(struct_name_arg) {
+ if let Some(struct_name) = struct_name_arg.to_atom() {
let fields: Vec<Atom> = match self.machine_st.try_from_list(fields_reg, stub_gen) {
Ok(addrs) => {
let mut args = Vec::new();
Err(e) => return Err(e),
};
self.foreign_function_table
- .define_struct(&struct_name.as_str(), fields)
+ .define_struct(struct_name, fields)
.map_err(|err| {
- let ffi_error = self.machine_st.ffi_error(err, struct_name_arg);
+ let ffi_error = self.machine_st.ffi_error(err);
self.machine_st.error_form(ffi_error, stub_gen())
})?;
- return Ok(());
+ Ok(())
+ } else {
+ let err = self.machine_st.type_error(ValidType::Atom, struct_name_arg);
+ Err(self.machine_st.error_form(err, stub_gen()))
}
- self.machine_st.fail = true;
- Ok(())
}
#[cfg(not(feature = "ffi"))]
) {
Ok(value) => value,
Err(ffi_error) => {
- let machine_error = self.machine_st.ffi_error(ffi_error, ffi_type_arg);
+ let machine_error = self.machine_st.ffi_error(ffi_error);
return Err(self.machine_st.error_form(machine_error, stub_gen()));
}
};
.foreign_function_table
.read_ptr(ffi_type, ptr, &mut self.machine_st.arena)
.map_err(|ffi_error| {
- let machine_error = self.machine_st.ffi_error(ffi_error, ffi_type_arg);
+ let machine_error = self.machine_st.ffi_error(ffi_error);
self.machine_st.error_form(machine_error, stub_gen())
})?;
{
Ok(value) => value,
Err(ffi_error) => {
- let machine_error = self.machine_st.ffi_error(ffi_error, ffi_type_arg);
+ let machine_error = self.machine_st.ffi_error(ffi_error);
return Err(self.machine_st.error_form(machine_error, stub_gen()));
}
}