use rustc::ty::{self, Ty};
-use rustc::ty::layout::{self, Align, LayoutOf};
+use rustc::ty::layout::{self, Align, LayoutOf, Size};
use rustc::hir::def_id::{DefId, CRATE_DEF_INDEX};
use rustc::mir;
use rustc_data_structures::indexed_vec::Idx;
.val;
let (discr_dest, discr) = ecx.place_field(dest, mir::Field::new(0), layout)?;
- ecx.write_primval(discr_dest, PrimVal::Bytes(discr_val), discr.ty)?;
+ ecx.write_scalar(discr_dest, Scalar::from_u128(discr_val), discr.ty)?;
}
layout::Variants::NicheFilling {
dataful_variant,
ecx.place_field(dest, mir::Field::new(0), layout)?;
let niche_value = ((variant_index - niche_variants.start()) as u128)
.wrapping_add(niche_start);
- ecx.write_primval(niche_dest, PrimVal::Bytes(niche_value), niche.ty)?;
+ ecx.write_scalar(niche_dest, Scalar::from_u128(niche_value), niche.ty)?;
}
}
}
_ => {}
}
+ if self.tcx.lang_items().align_offset_fn() == Some(instance.def.def_id()) {
+ // FIXME: return a real value in case the target allocation has an
+ // alignment bigger than the one requested
+ let n = u128::max_value();
+ let amt = 128 - self.memory.pointer_size().bytes() * 8;
+ let (dest, return_to_block) = destination.unwrap();
+ let ty = self.tcx.types.usize;
+ self.write_scalar(dest, Scalar::from_u128((n << amt) >> amt), ty)?;
+ self.goto_block(return_to_block);
+ return Ok(true);
+ }
+
let mir = match self.load_mir(instance.def) {
Ok(mir) => mir,
Err(EvalError { kind: EvalErrorKind::NoMirFor(path), .. }) => {
match &link_name[..] {
"malloc" => {
- let size = self.value_to_primval(args[0])?.to_u64()?;
+ let size = self.value_to_scalar(args[0])?.to_usize(self)?;
if size == 0 {
self.write_null(dest, dest_ty)?;
} else {
let align = self.tcx.data_layout.pointer_align;
- let ptr = self.memory.allocate(size, align, Some(MemoryKind::C.into()))?;
- self.write_primval(dest, PrimVal::Ptr(ptr), dest_ty)?;
+ let ptr = self.memory.allocate(Size::from_bytes(size), align, MemoryKind::C.into())?;
+ self.write_scalar(dest, Scalar::Ptr(ptr), dest_ty)?;
}
}
//
// libc::syscall(NR_GETRANDOM, buf.as_mut_ptr(), buf.len(), GRND_NONBLOCK)
// is called if a `HashMap` is created the regular way.
- match self.value_to_primval(args[0])?.to_u64()? {
+ match self.value_to_scalar(args[0])?.to_usize(self)? {
318 | 511 => {
return err!(Unimplemented(
"miri does not support random number generators".to_owned(),
)?;
let mut args = self.frame().mir.args_iter();
- let arg_local = args.next().ok_or(
+ let arg_local = args.next().ok_or_else(||
EvalErrorKind::AbiViolation(
"Argument to __rust_maybe_catch_panic does not take enough arguments."
.to_owned(),
"memcmp" => {
let left = self.into_ptr(args[0].value)?;
let right = self.into_ptr(args[1].value)?;
- let n = self.value_to_primval(args[2])?.to_u64()?;
+ let n = Size::from_bytes(self.value_to_scalar(args[2])?.to_usize(self)?);
let result = {
let left_bytes = self.memory.read_bytes(left, n)?;
use std::cmp::Ordering::*;
match left_bytes.cmp(right_bytes) {
- Less => -1i8,
+ Less => -1i32,
Equal => 0,
Greater => 1,
}
};
- self.write_primval(
+ self.write_scalar(
dest,
- PrimVal::Bytes(result as u128),
+ Scalar::from_i32(result),
dest_ty,
)?;
}
"memrchr" => {
let ptr = self.into_ptr(args[0].value)?;
- let val = self.value_to_primval(args[1])?.to_u64()? as u8;
- let num = self.value_to_primval(args[2])?.to_u64()?;
- if let Some(idx) = self.memory.read_bytes(ptr, num)?.iter().rev().position(
+ let val = self.value_to_scalar(args[1])?.to_bytes()? as u8;
+ let num = self.value_to_scalar(args[2])?.to_usize(self)?;
+ if let Some(idx) = self.memory.read_bytes(ptr, Size::from_bytes(num))?.iter().rev().position(
|&c| c == val,
)
{
- let new_ptr = ptr.offset(num - idx as u64 - 1, &self)?;
+ let new_ptr = ptr.ptr_offset(Size::from_bytes(num - idx as u64 - 1), &self)?;
self.write_ptr(dest, new_ptr, dest_ty)?;
} else {
self.write_null(dest, dest_ty)?;
"memchr" => {
let ptr = self.into_ptr(args[0].value)?;
- let val = self.value_to_primval(args[1])?.to_u64()? as u8;
- let num = self.value_to_primval(args[2])?.to_u64()?;
- if let Some(idx) = self.memory.read_bytes(ptr, num)?.iter().position(
+ let val = self.value_to_scalar(args[1])?.to_bytes()? as u8;
+ let num = self.value_to_scalar(args[2])?.to_usize(self)?;
+ if let Some(idx) = self.memory.read_bytes(ptr, Size::from_bytes(num))?.iter().position(
|&c| c == val,
)
{
- let new_ptr = ptr.offset(idx as u64, &self)?;
+ let new_ptr = ptr.ptr_offset(Size::from_bytes(idx as u64), &self)?;
self.write_ptr(dest, new_ptr, dest_ty)?;
} else {
self.write_null(dest, dest_ty)?;
let name_ptr = self.into_ptr(args[0].value)?.to_ptr()?;
let name = self.memory.read_c_str(name_ptr)?;
match self.machine.env_vars.get(name) {
- Some(&var) => PrimVal::Ptr(var),
- None => PrimVal::Bytes(0),
+ Some(&var) => Scalar::Ptr(var),
+ None => Scalar::null(),
}
};
- self.write_primval(dest, result, dest_ty)?;
+ self.write_scalar(dest, result, dest_ty)?;
}
"unsetenv" => {
}
self.write_null(dest, dest_ty)?;
} else {
- self.write_primval(dest, PrimVal::from_i128(-1), dest_ty)?;
+ self.write_scalar(dest, Scalar::from_i128(-1), dest_ty)?;
}
}
if let Some((name, value)) = new {
// +1 for the null terminator
let value_copy = self.memory.allocate(
- (value.len() + 1) as u64,
+ Size::from_bytes((value.len() + 1) as u64),
Align::from_bytes(1, 1).unwrap(),
- Some(MemoryKind::Env.into()),
+ MemoryKind::Env.into(),
)?;
self.memory.write_bytes(value_copy.into(), &value)?;
- let trailing_zero_ptr = value_copy.offset(value.len() as u64, &self)?.into();
+ let trailing_zero_ptr = value_copy.offset(Size::from_bytes(value.len() as u64), &self)?.into();
self.memory.write_bytes(trailing_zero_ptr, &[0])?;
if let Some(var) = self.machine.env_vars.insert(
name.to_owned(),
}
self.write_null(dest, dest_ty)?;
} else {
- self.write_primval(dest, PrimVal::from_i128(-1), dest_ty)?;
+ self.write_scalar(dest, Scalar::from_i128(-1), dest_ty)?;
}
}
"write" => {
- let fd = self.value_to_primval(args[0])?.to_u64()?;
+ let fd = self.value_to_scalar(args[0])?.to_bytes()?;
let buf = self.into_ptr(args[1].value)?;
- let n = self.value_to_primval(args[2])?.to_u64()?;
+ let n = self.value_to_scalar(args[2])?.to_bytes()? as u64;
trace!("Called write({:?}, {:?}, {:?})", fd, buf, n);
let result = if fd == 1 || fd == 2 {
// stdout/stderr
use std::io::{self, Write};
- let buf_cont = self.memory.read_bytes(buf, n)?;
+ let buf_cont = self.memory.read_bytes(buf, Size::from_bytes(n))?;
let res = if fd == 1 {
io::stdout().write(buf_cont)
} else {
io::stderr().write(buf_cont)
};
match res {
- Ok(n) => n as isize,
+ Ok(n) => n as i64,
Err(_) => -1,
}
} else {
warn!("Ignored output to FD {}", fd);
- n as isize // pretend it all went well
+ n as i64 // pretend it all went well
}; // now result is the value we return back to the program
- self.write_primval(
+ let ptr_size = self.memory.pointer_size();
+ self.write_scalar(
dest,
- PrimVal::Bytes(result as u128),
+ Scalar::from_isize(result, ptr_size),
dest_ty,
)?;
}
"strlen" => {
let ptr = self.into_ptr(args[0].value)?.to_ptr()?;
let n = self.memory.read_c_str(ptr)?.len();
- self.write_primval(dest, PrimVal::Bytes(n as u128), dest_ty)?;
+ let ptr_size = self.memory.pointer_size();
+ self.write_scalar(dest, Scalar::from_usize(n as u64, ptr_size), dest_ty)?;
}
// Some things needed for sys::thread initialization to go through
"signal" | "sigaction" | "sigaltstack" => {
- self.write_primval(dest, PrimVal::Bytes(0), dest_ty)?;
+ self.write_scalar(dest, Scalar::null(), dest_ty)?;
}
"sysconf" => {
- let name = self.value_to_primval(args[0])?.to_u64()?;
+ let name = self.value_to_scalar(args[0])?.to_usize(self)?;
trace!("sysconf() called with name {}", name);
// cache the sysconf integers via miri's global cache
let paths = &[
- (&["libc", "_SC_PAGESIZE"], PrimVal::Bytes(4096)),
- (&["libc", "_SC_GETPW_R_SIZE_MAX"], PrimVal::from_i128(-1)),
+ (&["libc", "_SC_PAGESIZE"], Scalar::from_i128(4096)),
+ (&["libc", "_SC_GETPW_R_SIZE_MAX"], Scalar::from_i128(-1)),
];
let mut result = None;
for &(path, path_value) in paths {
promoted: None,
};
let const_val = self.const_eval(cid)?;
- let value = const_val.val.unwrap_u64();
+ let value = const_val.unwrap_usize(self.tcx.tcx);
if value == name {
result = Some(path_value);
break;
}
}
if let Some(result) = result {
- self.write_primval(dest, result, dest_ty)?;
+ self.write_scalar(dest, result, dest_ty)?;
} else {
return err!(Unimplemented(
format!("Unimplemented sysconf name: {}", name),
let key_align = self.layout_of(args[0].ty)?.align;
// Extract the function type out of the signature (that seems easier than constructing it ourselves...)
- let dtor = match self.into_ptr(args[1].value)?.into_inner_primval() {
- PrimVal::Ptr(dtor_ptr) => Some(self.memory.get_fn(dtor_ptr)?),
- PrimVal::Bytes(0) => None,
- PrimVal::Bytes(_) => return err!(ReadBytesAsPointer),
- PrimVal::Undef => return err!(ReadUndefBytes),
+ let dtor = match self.into_ptr(args[1].value)? {
+ Scalar::Ptr(dtor_ptr) => Some(self.memory.get_fn(dtor_ptr)?),
+ Scalar::Bits { defined: 0, .. } => return err!(ReadUndefBytes),
+ Scalar::Bits { bits: 0, .. } => None,
+ Scalar::Bits { .. } => return err!(ReadBytesAsPointer),
};
// Figure out how large a pthread TLS key actually is. This is libc::pthread_key_t.
let key_type = args[0].ty.builtin_deref(true)
- .ok_or(EvalErrorKind::AbiViolation("Wrong signature used for pthread_key_create: First argument must be a raw pointer.".to_owned()))?.ty;
+ .ok_or_else(|| EvalErrorKind::AbiViolation("Wrong signature used for pthread_key_create: First argument must be a raw pointer.".to_owned()))?.ty;
let key_size = self.layout_of(key_type)?.size;
// Create key and write it into the memory where key_ptr wants it
if key_size.bits() < 128 && key >= (1u128 << key_size.bits() as u128) {
return err!(OutOfTls);
}
- self.memory.write_primval(
+ self.memory.write_scalar(
key_ptr,
key_align,
- PrimVal::Bytes(key),
- key_size.bytes(),
+ Scalar::from_u128(key),
+ key_size,
false,
)?;
self.write_null(dest, dest_ty)?;
}
"pthread_key_delete" => {
- // The conversion into TlsKey here is a little fishy, but should work as long as usize >= libc::pthread_key_t
- let key = self.value_to_primval(args[0])?.to_u64()? as TlsKey;
+ let key = self.value_to_scalar(args[0])?.to_bytes()?;
self.memory.delete_tls_key(key)?;
// Return success (0)
self.write_null(dest, dest_ty)?;
}
"pthread_getspecific" => {
- // The conversion into TlsKey here is a little fishy, but should work as long as usize >= libc::pthread_key_t
- let key = self.value_to_primval(args[0])?.to_u64()? as TlsKey;
+ let key = self.value_to_scalar(args[0])?.to_bytes()?;
let ptr = self.memory.load_tls(key)?;
self.write_ptr(dest, ptr, dest_ty)?;
}
"pthread_setspecific" => {
- // The conversion into TlsKey here is a little fishy, but should work as long as usize >= libc::pthread_key_t
- let key = self.value_to_primval(args[0])?.to_u64()? as TlsKey;
+ let key = self.value_to_scalar(args[0])?.to_bytes()?;
let new_ptr = self.into_ptr(args[1].value)?;
self.memory.store_tls(key, new_ptr)?;
// Return success (0)
self.write_null(dest, dest_ty)?;
}
+
"_tlv_atexit" => {
- // TODO: handle it
- }
+ return err!(Unimplemented("can't interpret with full mir for osx target".to_owned()));
+ },
// Stub out all the other pthread calls to just return 0
link_name if link_name.starts_with("pthread_") => {
self.write_null(dest, dest_ty)?;
}
+ "mmap" => {
+ // This is a horrible hack, but well... the guard page mechanism calls mmap and expects a particular return value, so we give it that value
+ let addr = self.into_ptr(args[0].value)?;
+ self.write_ptr(dest, addr, dest_ty)?;
+ }
+
_ => {
return err!(Unimplemented(
format!("can't call C ABI function: {}", link_name),
match &path[..] {
// Allocators are magic. They have no MIR, even when the rest of libstd does.
"alloc::alloc::::__rust_alloc" => {
- let size = self.value_to_primval(args[0])?.to_u64()?;
- let align = self.value_to_primval(args[1])?.to_u64()?;
+ let size = self.value_to_scalar(args[0])?.to_usize(self)?;
+ let align = self.value_to_scalar(args[1])?.to_usize(self)?;
if size == 0 {
return err!(HeapAllocZeroBytes);
}
if !align.is_power_of_two() {
return err!(HeapAllocNonPowerOfTwoAlignment(align));
}
- let ptr = self.memory.allocate(size,
+ let ptr = self.memory.allocate(Size::from_bytes(size),
Align::from_bytes(align, align).unwrap(),
- Some(MemoryKind::Rust.into()))?;
- self.write_primval(dest, PrimVal::Ptr(ptr), dest_ty)?;
+ MemoryKind::Rust.into())?;
+ self.write_scalar(dest, Scalar::Ptr(ptr), dest_ty)?;
}
"alloc::alloc::::__rust_alloc_zeroed" => {
- let size = self.value_to_primval(args[0])?.to_u64()?;
- let align = self.value_to_primval(args[1])?.to_u64()?;
+ let size = self.value_to_scalar(args[0])?.to_usize(self)?;
+ let align = self.value_to_scalar(args[1])?.to_usize(self)?;
if size == 0 {
return err!(HeapAllocZeroBytes);
}
if !align.is_power_of_two() {
return err!(HeapAllocNonPowerOfTwoAlignment(align));
}
- let ptr = self.memory.allocate(size,
+ let ptr = self.memory.allocate(Size::from_bytes(size),
Align::from_bytes(align, align).unwrap(),
- Some(MemoryKind::Rust.into()))?;
- self.memory.write_repeat(ptr.into(), 0, size)?;
- self.write_primval(dest, PrimVal::Ptr(ptr), dest_ty)?;
+ MemoryKind::Rust.into())?;
+ self.memory.write_repeat(ptr.into(), 0, Size::from_bytes(size))?;
+ self.write_scalar(dest, Scalar::Ptr(ptr), dest_ty)?;
}
"alloc::alloc::::__rust_dealloc" => {
let ptr = self.into_ptr(args[0].value)?.to_ptr()?;
- let old_size = self.value_to_primval(args[1])?.to_u64()?;
- let align = self.value_to_primval(args[2])?.to_u64()?;
+ let old_size = self.value_to_scalar(args[1])?.to_usize(self)?;
+ let align = self.value_to_scalar(args[2])?.to_usize(self)?;
if old_size == 0 {
return err!(HeapAllocZeroBytes);
}
}
self.memory.deallocate(
ptr,
- Some((old_size, Align::from_bytes(align, align).unwrap())),
+ Some((Size::from_bytes(old_size), Align::from_bytes(align, align).unwrap())),
MemoryKind::Rust.into(),
)?;
}
"alloc::alloc::::__rust_realloc" => {
let ptr = self.into_ptr(args[0].value)?.to_ptr()?;
- let old_size = self.value_to_primval(args[1])?.to_u64()?;
- let align = self.value_to_primval(args[2])?.to_u64()?;
- let new_size = self.value_to_primval(args[3])?.to_u64()?;
+ let old_size = self.value_to_scalar(args[1])?.to_usize(self)?;
+ let align = self.value_to_scalar(args[2])?.to_usize(self)?;
+ let new_size = self.value_to_scalar(args[3])?.to_usize(self)?;
if old_size == 0 || new_size == 0 {
return err!(HeapAllocZeroBytes);
}
}
let new_ptr = self.memory.reallocate(
ptr,
- old_size,
+ Size::from_bytes(old_size),
Align::from_bytes(align, align).unwrap(),
- new_size,
+ Size::from_bytes(new_size),
Align::from_bytes(align, align).unwrap(),
MemoryKind::Rust.into(),
)?;
- self.write_primval(dest, PrimVal::Ptr(new_ptr), dest_ty)?;
+ self.write_scalar(dest, Scalar::Ptr(new_ptr), dest_ty)?;
}
// A Rust function is missing, which means we are running with MIR missing for libstd (or other dependencies).
"std::rt::panicking" => {
// we abort on panic -> `std::rt::panicking` always returns false
let bool = self.tcx.types.bool;
- self.write_primval(dest, PrimVal::from_bool(false), bool)?;
+ self.write_scalar(dest, Scalar::from_bool(false), bool)?;
}
"std::sys::imp::c::::AddVectoredExceptionHandler" |
"std::sys::imp::c::::SetThreadStackGuarantee" => {
let usize = self.tcx.types.usize;
// any non zero value works for the stdlib. This is just used for stackoverflows anyway
- self.write_primval(dest, PrimVal::Bytes(1), usize)?;
+ self.write_scalar(dest, Scalar::from_u128(1), usize)?;
},
_ => return err!(NoMirFor(path)),
}
// current frame.
self.dump_local(dest);
self.goto_block(dest_block);
- return Ok(());
+ Ok(())
}
fn write_null(&mut self, dest: Place, dest_ty: Ty<'tcx>) -> EvalResult<'tcx> {
- self.write_primval(dest, PrimVal::Bytes(0), dest_ty)
+ self.write_scalar(dest, Scalar::null(), dest_ty)
}
}