2019-06-30 16:03:13 +02:00
|
|
|
use rustc::mir;
|
|
|
|
|
|
|
|
use crate::*;
|
|
|
|
|
|
|
|
#[derive(Debug, Copy, Clone)]
|
|
|
|
pub enum Dlsym {
|
|
|
|
GetEntropy,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Dlsym {
|
2019-06-30 16:43:05 +02:00
|
|
|
// Returns an error for unsupported symbols, and None if this symbol
|
|
|
|
// should become a NULL pointer (pretend it does not exist).
|
|
|
|
pub fn from_str(name: &str) -> InterpResult<'static, Option<Dlsym>> {
|
2019-06-30 16:03:13 +02:00
|
|
|
use self::Dlsym::*;
|
2019-06-30 16:43:05 +02:00
|
|
|
Ok(match name {
|
|
|
|
"getentropy" => Some(GetEntropy),
|
|
|
|
"__pthread_get_minstack" => None,
|
|
|
|
_ =>
|
2019-08-03 20:31:33 +02:00
|
|
|
throw_unsup_format!("Unsupported dlsym: {}", name),
|
2019-06-30 16:03:13 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'mir, 'tcx> EvalContextExt<'mir, 'tcx> for crate::MiriEvalContext<'mir, 'tcx> {}
|
|
|
|
pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriEvalContextExt<'mir, 'tcx> {
|
|
|
|
fn call_dlsym(
|
|
|
|
&mut self,
|
|
|
|
dlsym: Dlsym,
|
|
|
|
args: &[OpTy<'tcx, Tag>],
|
|
|
|
dest: Option<PlaceTy<'tcx, Tag>>,
|
|
|
|
ret: Option<mir::BasicBlock>,
|
|
|
|
) -> InterpResult<'tcx> {
|
|
|
|
use self::Dlsym::*;
|
|
|
|
|
|
|
|
let this = self.eval_context_mut();
|
|
|
|
|
|
|
|
let dest = dest.expect("we don't support any diverging dlsym");
|
|
|
|
let ret = ret.expect("dest is `Some` but ret is `None`");
|
2019-06-30 16:43:05 +02:00
|
|
|
|
2019-06-30 16:03:13 +02:00
|
|
|
match dlsym {
|
|
|
|
GetEntropy => {
|
|
|
|
let ptr = this.read_scalar(args[0])?.not_undef()?;
|
|
|
|
let len = this.read_scalar(args[1])?.to_usize(this)?;
|
|
|
|
this.gen_random(len as usize, ptr)?;
|
|
|
|
this.write_null(dest)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.goto_block(Some(ret))?;
|
|
|
|
this.dump_place(*dest);
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|