2016-12-10 18:23:07 -06:00
|
|
|
#![allow(unknown_lints)]
|
|
|
|
#![allow(float_cmp)]
|
|
|
|
|
|
|
|
use std::mem::transmute;
|
|
|
|
|
|
|
|
use error::{EvalError, EvalResult};
|
|
|
|
use memory::{AllocId, Memory, Pointer};
|
|
|
|
|
|
|
|
pub(super) fn bits_to_f32(bits: u64) -> f32 {
|
|
|
|
unsafe { transmute::<u32, f32>(bits as u32) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn bits_to_f64(bits: u64) -> f64 {
|
|
|
|
unsafe { transmute::<u64, f64>(bits) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn f32_to_bits(f: f32) -> u64 {
|
|
|
|
unsafe { transmute::<f32, u32>(f) as u64 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn f64_to_bits(f: f64) -> u64 {
|
|
|
|
unsafe { transmute::<f64, u64>(f) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn bits_to_bool(n: u64) -> bool {
|
|
|
|
// FIXME(solson): Can we reach here due to user error?
|
|
|
|
debug_assert!(n == 0 || n == 1, "bits interpreted as bool were {}", n);
|
|
|
|
n & 1 == 1
|
|
|
|
}
|
2016-09-23 03:27:14 -05:00
|
|
|
|
|
|
|
/// A `Value` represents a single self-contained Rust value.
|
|
|
|
///
|
|
|
|
/// A `Value` can either refer to a block of memory inside an allocation (`ByRef`) or to a primitve
|
|
|
|
/// value held directly, outside of any allocation (`ByVal`).
|
|
|
|
///
|
|
|
|
/// For optimization of a few very common cases, there is also a representation for a pair of
|
|
|
|
/// primitive values (`ByValPair`). It allows Miri to avoid making allocations for checked binary
|
|
|
|
/// operations and fat pointers. This idea was taken from rustc's trans.
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
2016-10-14 04:31:45 -05:00
|
|
|
pub enum Value {
|
2016-09-23 03:27:14 -05:00
|
|
|
ByRef(Pointer),
|
|
|
|
ByVal(PrimVal),
|
2016-09-26 10:49:30 -05:00
|
|
|
ByValPair(PrimVal, PrimVal),
|
2016-09-23 03:27:14 -05:00
|
|
|
}
|
|
|
|
|
2016-12-16 01:55:00 -06:00
|
|
|
/// A `PrimVal` represents an immediate, primitive value existing outside of a
|
|
|
|
/// `memory::Allocation`. It is in many ways like a small chunk of a `Allocation`, up to 8 bytes in
|
|
|
|
/// size. Like a range of bytes in an `Allocation`, a `PrimVal` can either represent the raw bytes
|
|
|
|
/// of a simple value, a pointer into another `Allocation`, or be undefined.
|
2016-12-10 18:23:07 -06:00
|
|
|
#[derive(Clone, Copy, Debug, PartialEq)]
|
2016-12-16 01:40:45 -06:00
|
|
|
pub enum PrimVal {
|
2016-12-16 01:55:00 -06:00
|
|
|
/// The raw bytes of a simple value.
|
2016-12-16 01:40:45 -06:00
|
|
|
Bytes(u64),
|
2016-12-10 18:23:07 -06:00
|
|
|
|
2016-12-16 01:55:00 -06:00
|
|
|
/// A pointer into an `Allocation`. An `Allocation` in the `memory` module has a list of
|
|
|
|
/// relocations, but a `PrimVal` is only large enough to contain one, so we just represent the
|
|
|
|
/// relocation and its associated offset together as a `Pointer` here.
|
|
|
|
Ptr(Pointer),
|
2016-12-16 01:40:45 -06:00
|
|
|
|
2016-12-16 01:55:00 -06:00
|
|
|
/// An undefined `PrimVal`, for representing values that aren't safe to examine, but are safe
|
|
|
|
/// to copy around, just like undefined bytes in an `Allocation`.
|
2016-12-16 01:40:45 -06:00
|
|
|
Undefined,
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq)]
|
|
|
|
pub enum PrimValKind {
|
|
|
|
I8, I16, I32, I64,
|
|
|
|
U8, U16, U32, U64,
|
|
|
|
F32, F64,
|
|
|
|
Bool,
|
|
|
|
Char,
|
|
|
|
Ptr,
|
|
|
|
FnPtr,
|
|
|
|
}
|
|
|
|
|
2016-10-14 04:31:45 -05:00
|
|
|
impl<'a, 'tcx: 'a> Value {
|
|
|
|
pub(super) fn read_ptr(&self, mem: &Memory<'a, 'tcx>) -> EvalResult<'tcx, Pointer> {
|
2016-09-23 03:27:14 -05:00
|
|
|
use self::Value::*;
|
|
|
|
match *self {
|
|
|
|
ByRef(ptr) => mem.read_ptr(ptr),
|
2016-11-15 07:12:49 -06:00
|
|
|
ByVal(ptr) | ByValPair(ptr, _) => Ok(ptr.to_ptr()),
|
2016-09-26 10:49:30 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-16 03:12:26 -05:00
|
|
|
pub(super) fn expect_ptr_vtable_pair(
|
|
|
|
&self,
|
|
|
|
mem: &Memory<'a, 'tcx>
|
|
|
|
) -> EvalResult<'tcx, (Pointer, Pointer)> {
|
2016-09-23 03:27:14 -05:00
|
|
|
use self::Value::*;
|
|
|
|
match *self {
|
2016-11-15 07:12:49 -06:00
|
|
|
ByRef(ref_ptr) => {
|
|
|
|
let ptr = mem.read_ptr(ref_ptr)?;
|
2016-11-18 05:55:14 -06:00
|
|
|
let vtable = mem.read_ptr(ref_ptr.offset(mem.pointer_size()))?;
|
2016-10-16 03:12:26 -05:00
|
|
|
Ok((ptr, vtable))
|
|
|
|
}
|
2016-10-20 05:42:19 -05:00
|
|
|
|
2016-11-15 07:12:49 -06:00
|
|
|
ByValPair(ptr, vtable) => Ok((ptr.to_ptr(), vtable.to_ptr())),
|
2016-10-20 05:42:19 -05:00
|
|
|
|
2016-10-16 03:12:26 -05:00
|
|
|
_ => bug!("expected ptr and vtable, got {:?}", self),
|
2016-09-23 03:27:14 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-15 07:12:49 -06:00
|
|
|
pub(super) fn expect_slice(&self, mem: &Memory<'a, 'tcx>) -> EvalResult<'tcx, (Pointer, u64)> {
|
2016-09-23 03:27:14 -05:00
|
|
|
use self::Value::*;
|
|
|
|
match *self {
|
2016-11-15 07:12:49 -06:00
|
|
|
ByRef(ref_ptr) => {
|
|
|
|
let ptr = mem.read_ptr(ref_ptr)?;
|
2016-11-18 05:55:14 -06:00
|
|
|
let len = mem.read_usize(ref_ptr.offset(mem.pointer_size()))?;
|
2016-11-15 07:12:49 -06:00
|
|
|
Ok((ptr, len))
|
|
|
|
},
|
|
|
|
ByValPair(ptr, val) => {
|
|
|
|
Ok((ptr.to_ptr(), val.try_as_uint()?))
|
|
|
|
},
|
2016-09-23 03:27:14 -05:00
|
|
|
_ => unimplemented!(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-10 18:23:07 -06:00
|
|
|
|
|
|
|
impl PrimVal {
|
2016-12-16 01:40:45 -06:00
|
|
|
// FIXME(solson): Remove this. It's a temporary function to aid refactoring, but it shouldn't
|
|
|
|
// stick around with this name.
|
|
|
|
pub fn bits(&self) -> u64 {
|
|
|
|
match *self {
|
|
|
|
PrimVal::Bytes(b) => b,
|
2016-12-16 01:55:00 -06:00
|
|
|
PrimVal::Ptr(p) => p.offset,
|
2016-12-16 01:40:45 -06:00
|
|
|
PrimVal::Undefined => panic!(".bits()() on PrimVal::Undefined"),
|
|
|
|
}
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
2016-12-16 01:40:45 -06:00
|
|
|
// FIXME(solson): Remove this. It's a temporary function to aid refactoring, but it shouldn't
|
|
|
|
// stick around with this name.
|
|
|
|
pub fn relocation(&self) -> Option<AllocId> {
|
2016-12-16 01:55:00 -06:00
|
|
|
if let PrimVal::Ptr(ref p) = *self {
|
2016-12-16 01:40:45 -06:00
|
|
|
Some(p.alloc_id)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_bool(b: bool) -> Self {
|
2016-12-16 01:40:45 -06:00
|
|
|
PrimVal::Bytes(b as u64)
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_char(c: char) -> Self {
|
2016-12-16 01:40:45 -06:00
|
|
|
PrimVal::Bytes(c as u64)
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_f32(f: f32) -> Self {
|
2016-12-16 01:40:45 -06:00
|
|
|
PrimVal::Bytes(f32_to_bits(f))
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_f64(f: f64) -> Self {
|
2016-12-16 01:40:45 -06:00
|
|
|
PrimVal::Bytes(f64_to_bits(f))
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_uint(n: u64) -> Self {
|
2016-12-16 01:40:45 -06:00
|
|
|
PrimVal::Bytes(n)
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_int(n: i64) -> Self {
|
2016-12-16 01:40:45 -06:00
|
|
|
PrimVal::Bytes(n as u64)
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn to_f32(self) -> f32 {
|
2016-12-16 01:40:45 -06:00
|
|
|
assert!(self.relocation().is_none());
|
|
|
|
bits_to_f32(self.bits())
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn to_f64(self) -> f64 {
|
2016-12-16 01:40:45 -06:00
|
|
|
assert!(self.relocation().is_none());
|
|
|
|
bits_to_f64(self.bits())
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn to_ptr(self) -> Pointer {
|
2016-12-16 01:40:45 -06:00
|
|
|
self.relocation().map(|alloc_id| {
|
|
|
|
Pointer::new(alloc_id, self.bits())
|
|
|
|
}).unwrap_or_else(|| Pointer::from_int(self.bits()))
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn try_as_uint<'tcx>(self) -> EvalResult<'tcx, u64> {
|
|
|
|
self.to_ptr().to_int()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn to_u64(self) -> u64 {
|
|
|
|
if let Some(ptr) = self.try_as_ptr() {
|
|
|
|
return ptr.to_int().expect("non abstract ptr") as u64;
|
|
|
|
}
|
2016-12-16 01:40:45 -06:00
|
|
|
self.bits()
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn to_i64(self) -> i64 {
|
|
|
|
if let Some(ptr) = self.try_as_ptr() {
|
|
|
|
return ptr.to_int().expect("non abstract ptr") as i64;
|
|
|
|
}
|
2016-12-16 01:40:45 -06:00
|
|
|
self.bits() as i64
|
2016-12-10 18:23:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn try_as_ptr(self) -> Option<Pointer> {
|
2016-12-16 01:40:45 -06:00
|
|
|
self.relocation().map(|alloc_id| {
|
|
|
|
Pointer::new(alloc_id, self.bits())
|
2016-12-10 18:23:07 -06:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn try_as_bool<'tcx>(self) -> EvalResult<'tcx, bool> {
|
2016-12-16 01:40:45 -06:00
|
|
|
match self.bits() {
|
2016-12-10 18:23:07 -06:00
|
|
|
0 => Ok(false),
|
|
|
|
1 => Ok(true),
|
|
|
|
_ => Err(EvalError::InvalidBool),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PrimValKind {
|
|
|
|
pub fn is_int(self) -> bool {
|
|
|
|
use self::PrimValKind::*;
|
|
|
|
match self {
|
|
|
|
I8 | I16 | I32 | I64 | U8 | U16 | U32 | U64 => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_uint_size(size: u64) -> Self {
|
|
|
|
match size {
|
|
|
|
1 => PrimValKind::U8,
|
|
|
|
2 => PrimValKind::U16,
|
|
|
|
4 => PrimValKind::U32,
|
|
|
|
8 => PrimValKind::U64,
|
|
|
|
_ => bug!("can't make uint with size {}", size),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_int_size(size: u64) -> Self {
|
|
|
|
match size {
|
|
|
|
1 => PrimValKind::I8,
|
|
|
|
2 => PrimValKind::I16,
|
|
|
|
4 => PrimValKind::I32,
|
|
|
|
8 => PrimValKind::I64,
|
|
|
|
_ => bug!("can't make int with size {}", size),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|