2018-05-01 11:13:22 -05:00
|
|
|
use mir;
|
2017-07-28 06:08:27 -05:00
|
|
|
use rustc::ty::Ty;
|
2017-12-06 08:03:24 -06:00
|
|
|
use rustc::ty::layout::LayoutOf;
|
2017-07-28 06:08:27 -05:00
|
|
|
|
2018-05-01 11:13:22 -05:00
|
|
|
use super::{Pointer, EvalResult, PrimVal, EvalContext};
|
|
|
|
|
2017-07-28 06:08:27 -05:00
|
|
|
pub trait EvalContextExt<'tcx> {
|
|
|
|
fn wrapping_pointer_offset(
|
|
|
|
&self,
|
|
|
|
ptr: Pointer,
|
|
|
|
pointee_ty: Ty<'tcx>,
|
|
|
|
offset: i64,
|
|
|
|
) -> EvalResult<'tcx, Pointer>;
|
|
|
|
|
|
|
|
fn pointer_offset(
|
|
|
|
&self,
|
|
|
|
ptr: Pointer,
|
|
|
|
pointee_ty: Ty<'tcx>,
|
|
|
|
offset: i64,
|
|
|
|
) -> EvalResult<'tcx, Pointer>;
|
|
|
|
}
|
|
|
|
|
2018-01-14 11:59:13 -06:00
|
|
|
impl<'a, 'mir, 'tcx> EvalContextExt<'tcx> for EvalContext<'a, 'mir, 'tcx, super::Evaluator<'tcx>> {
|
2017-07-28 06:08:27 -05:00
|
|
|
fn wrapping_pointer_offset(
|
|
|
|
&self,
|
|
|
|
ptr: Pointer,
|
|
|
|
pointee_ty: Ty<'tcx>,
|
|
|
|
offset: i64,
|
|
|
|
) -> EvalResult<'tcx, Pointer> {
|
|
|
|
// FIXME: assuming here that type size is < i64::max_value()
|
2017-12-06 08:03:24 -06:00
|
|
|
let pointee_size = self.layout_of(pointee_ty)?.size.bytes() as i64;
|
2017-07-28 06:08:27 -05:00
|
|
|
let offset = offset.overflowing_mul(pointee_size).0;
|
|
|
|
ptr.wrapping_signed_offset(offset, self)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn pointer_offset(
|
|
|
|
&self,
|
|
|
|
ptr: Pointer,
|
|
|
|
pointee_ty: Ty<'tcx>,
|
|
|
|
offset: i64,
|
|
|
|
) -> EvalResult<'tcx, Pointer> {
|
|
|
|
// This function raises an error if the offset moves the pointer outside of its allocation. We consider
|
|
|
|
// ZSTs their own huge allocation that doesn't overlap with anything (and nothing moves in there because the size is 0).
|
|
|
|
// We also consider the NULL pointer its own separate allocation, and all the remaining integers pointers their own
|
|
|
|
// allocation.
|
|
|
|
|
2017-08-10 10:48:38 -05:00
|
|
|
if ptr.is_null()? {
|
|
|
|
// NULL pointers must only be offset by 0
|
|
|
|
return if offset == 0 {
|
|
|
|
Ok(ptr)
|
|
|
|
} else {
|
|
|
|
err!(InvalidNullPointerUsage)
|
|
|
|
};
|
2017-07-28 06:08:27 -05:00
|
|
|
}
|
|
|
|
// FIXME: assuming here that type size is < i64::max_value()
|
2017-12-06 08:03:24 -06:00
|
|
|
let pointee_size = self.layout_of(pointee_ty)?.size.bytes() as i64;
|
2017-07-28 06:08:27 -05:00
|
|
|
return if let Some(offset) = offset.checked_mul(pointee_size) {
|
|
|
|
let ptr = ptr.signed_offset(offset, self)?;
|
|
|
|
// Do not do bounds-checking for integers; they can never alias a normal pointer anyway.
|
|
|
|
if let PrimVal::Ptr(ptr) = ptr.into_inner_primval() {
|
|
|
|
self.memory.check_bounds(ptr, false)?;
|
|
|
|
} else if ptr.is_null()? {
|
|
|
|
// We moved *to* a NULL pointer. That seems wrong, LLVM considers the NULL pointer its own small allocation. Reject this, for now.
|
2017-08-02 09:59:01 -05:00
|
|
|
return err!(InvalidNullPointerUsage);
|
2017-07-28 06:08:27 -05:00
|
|
|
}
|
|
|
|
Ok(ptr)
|
|
|
|
} else {
|
2018-05-01 11:13:22 -05:00
|
|
|
err!(Overflow(mir::BinOp::Mul))
|
2017-08-10 10:48:38 -05:00
|
|
|
};
|
2017-07-28 06:08:27 -05:00
|
|
|
}
|
|
|
|
}
|