openethereum/src/evm/jit.rs

201 lines
4.8 KiB
Rust
Raw Normal View History

2015-12-23 13:02:01 +01:00
use std::mem;
use evmjit;
use util::hash::*;
use util::uint::*;
use util::bytes::*;
2015-12-24 01:07:46 +01:00
use util::sha3::*;
2015-12-23 13:02:01 +01:00
use evm;
2015-12-23 13:49:45 +01:00
/// Should be used to convert jit types to ethcore
pub trait FromJit<T>: Sized {
fn from_jit(input: T) -> Self;
2015-12-23 13:02:01 +01:00
}
2015-12-23 13:49:45 +01:00
/// Should be used to covert ethcore types to jit
pub trait IntoJit<T> {
fn into_jit(self) -> T;
2015-12-23 13:02:01 +01:00
}
2015-12-23 13:49:45 +01:00
impl<'a> FromJit<&'a evmjit::I256> for U256 {
fn from_jit(input: &'a evmjit::I256) -> Self {
2015-12-23 13:02:01 +01:00
let mut res: U256 = unsafe { mem::uninitialized() };
res.0[0] = input.words[3];
res.0[1] = input.words[2];
res.0[2] = input.words[1];
res.0[3] = input.words[0];
res
}
}
2015-12-23 13:49:45 +01:00
impl<'a> FromJit<&'a evmjit::I256> for H256 {
fn from_jit(input: &'a evmjit::I256) -> Self {
2015-12-23 13:02:01 +01:00
let u = U256::from_jit(input);
H256::from(&u)
}
}
2015-12-23 13:49:45 +01:00
impl IntoJit<evmjit::I256> for U256 {
2015-12-23 13:02:01 +01:00
fn into_jit(self) -> evmjit::I256 {
let mut res: evmjit::I256 = unsafe { mem::uninitialized() };
res.words[0] = self.0[3];
res.words[1] = self.0[2];
res.words[2] = self.0[1];
res.words[3] = self.0[0];
res
}
}
2015-12-23 13:49:45 +01:00
impl IntoJit<evmjit::I256> for H256 {
2015-12-23 13:02:01 +01:00
fn into_jit(self) -> evmjit::I256 {
let mut ret = [0; 4];
for i in 0..self.bytes().len() {
let rev = self.bytes().len() - 1 - i;
let pos = i / 8;
ret[pos] += (self.bytes()[i] as u64) << (rev % 8) * 8;
}
evmjit::I256 { words: ret }
}
}
2015-12-24 01:07:46 +01:00
impl IntoJit<evmjit::I256> for Address {
fn into_jit(self) -> evmjit::I256 {
H256::from(self).into_jit()
}
}
impl IntoJit<evmjit::RuntimeDataHandle> for evm::RuntimeData {
fn into_jit(self) -> evmjit::RuntimeDataHandle {
let mut data = evmjit::RuntimeDataHandle::new();
data.gas = self.gas as i64;
data.gas_price = self.gas_price as i64;
data.call_data = self.call_data.as_ptr();
data.call_data_size = self.call_data.len() as u64;
mem::forget(self.call_data);
data.address = self.address.into_jit();
data.caller = self.caller.into_jit();
data.origin = self.origin.into_jit();
data.call_value = self.call_value.into_jit();
data.coinbase = self.coinbase.into_jit();
data.difficulty = self.difficulty.into_jit();
data.gas_limit = self.gas_limit.into_jit();
data.number = self.number;
data.timestamp = self.timestamp as i64;
data.code = self.code.as_ptr();
data.code_size = self.code.len() as u64;
data.code_hash = self.code.sha3().into_jit();
mem::forget(self.code);
data
}
}
2015-12-23 13:02:01 +01:00
pub struct EnvAdapter {
env: evm::Env
}
impl EnvAdapter {
pub fn new() -> EnvAdapter {
EnvAdapter {
env: evm::Env::new()
}
}
}
impl evmjit::Env for EnvAdapter {
fn sload(&self, index: *const evmjit::I256, out_value: *mut evmjit::I256) {
unsafe {
let i = H256::from_jit(&*index);
let o = self.env.sload(&i);
*out_value = o.into_jit();
}
}
fn sstore(&mut self, index: *const evmjit::I256, value: *const evmjit::I256) {
unsafe {
self.env.sstore(&H256::from_jit(&*index), &H256::from_jit(&*value));
}
}
fn balance(&self, _address: *const evmjit::I256, _out_value: *mut evmjit::I256) {
unimplemented!();
}
fn blockhash(&self, _number: *const evmjit::I256, _out_hash: *mut evmjit::I256) {
unimplemented!();
}
fn create(&mut self,
_io_gas: *mut u64,
_endowment: *const evmjit::I256,
_init_beg: *const u8,
_init_size: *const u64,
_address: *mut evmjit::I256) {
unimplemented!();
}
fn call(&mut self,
_io_gas: *mut u64,
_call_gas: *const u64,
_receive_address: *const evmjit::I256,
_value: *const evmjit::I256,
_in_beg: *const u8,
_in_size: *const u64,
_out_beg: *mut u8,
_out_size: *mut u64,
_code_address: evmjit::I256) -> bool {
unimplemented!();
}
fn log(&mut self,
_beg: *const u8,
_size: *const u64,
_topic1: *const evmjit::I256,
_topic2: *const evmjit::I256,
_topic3: *const evmjit::I256,
_topic4: *const evmjit::I256) {
unimplemented!();
}
fn extcode(&self, _address: *const evmjit::I256, _size: *mut u64) -> *const u8 {
unimplemented!();
}
}
2015-12-23 13:49:45 +01:00
#[cfg(test)]
mod tests {
use util::hash::*;
use util::uint::*;
use evmjit::{ContextHandle, RuntimeDataHandle, EnvHandle, ReturnCode};
use evm::*;
use evm::jit::{FromJit, IntoJit};
#[test]
fn test_to_and_from_u256() {
use std::str::FromStr;
let u = U256::from_str("d4e56740f876aef8c010b86a40d5f56745a118d0906a34e69aec8c0db1cb8fa3").unwrap();
let j = u.into_jit();
let u2 = U256::from_jit(&j);
assert_eq!(u, u2);
}
2015-12-23 13:02:01 +01:00
2015-12-23 13:49:45 +01:00
#[test]
fn test_to_and_from_h256() {
use std::str::FromStr;
2015-12-23 13:02:01 +01:00
2015-12-23 13:49:45 +01:00
let h = H256::from_str("d4e56740f876aef8c010b86a40d5f56745a118d0906a34e69aec8c0db1cb8fa3").unwrap();
let j = h.clone().into_jit();
let h2 = H256::from_jit(&j);
assert_eq!(h, h2);
}
2015-12-23 13:02:01 +01:00
2015-12-23 13:49:45 +01:00
#[test]
fn test_env_adapter() {
2015-12-24 01:07:46 +01:00
let mut data = RuntimeData::new();
data.code = vec![0x60, 0x00, 0x60, 0x00, 0x20, 0x60, 0x00, 0x55];
2015-12-23 13:49:45 +01:00
let env = EnvAdapter::new();
2015-12-24 01:07:46 +01:00
let mut context = ContextHandle::new(data.into_jit(), EnvHandle::new(env));
2015-12-23 13:49:45 +01:00
assert_eq!(context.exec(), ReturnCode::Stop);
}
2015-12-23 13:02:01 +01:00
}