Merge pull request #2018 from ethcore/builtin_trait
builtin trait refactoring
This commit is contained in:
		
						commit
						a88440ebae
					
				| @ -14,283 +14,350 @@ | |||||||
| // You should have received a copy of the GNU General Public License
 | // You should have received a copy of the GNU General Public License
 | ||||||
| // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | // along with Parity.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
| 
 | 
 | ||||||
| use crypto::sha2::Sha256; | use crypto::sha2::Sha256 as Sha256Digest; | ||||||
| use crypto::ripemd160::Ripemd160; | use crypto::ripemd160::Ripemd160 as Ripemd160Digest; | ||||||
| use crypto::digest::Digest; | use crypto::digest::Digest; | ||||||
| use util::*; | use util::*; | ||||||
| use ethkey::{Signature, recover}; | use ethkey::{Signature, recover as ec_recover}; | ||||||
| use ethjson; | use ethjson; | ||||||
| 
 | 
 | ||||||
| /// Definition of a contract whose implementation is built-in.
 | /// Native implementation of a built-in contract.
 | ||||||
| pub struct Builtin { | pub trait Impl: Send + Sync { | ||||||
| 	/// The gas cost of running this built-in for the given size of input data.
 | 	/// execute this built-in on the given input, writing to the given output.
 | ||||||
| 	pub cost: Box<Fn(usize) -> U256>,	// TODO: U256 should be bignum.
 | 	fn execute(&self, input: &[u8], out: &mut [u8]); | ||||||
| 	/// Run this built-in function with the input being the first argument and the output
 |  | ||||||
| 	/// being placed into the second.
 |  | ||||||
| 	pub execute: Box<Fn(&[u8], &mut [u8])>, |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Rust does not mark closurer that do not capture as Sync
 | /// A gas pricing scheme for built-in contracts.
 | ||||||
| // We promise that all builtins are thread safe since they only operate on given input.
 | pub trait Pricer: Send + Sync { | ||||||
| unsafe impl Sync for Builtin {} | 	/// The gas cost of running this built-in for the given size of input data.
 | ||||||
| unsafe impl Send for Builtin {} | 	fn cost(&self, in_size: usize) -> U256; | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| impl fmt::Debug for Builtin { | /// A linear pricing model. This computes a price using a base cost and a cost per-word.
 | ||||||
| 	fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | struct Linear { | ||||||
| 		write!(f, "<Builtin>") | 	base: usize, | ||||||
|  | 	word: usize, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Pricer for Linear { | ||||||
|  | 	fn cost(&self, in_size: usize) -> U256 { | ||||||
|  | 		U256::from(self.base) + U256::from(self.word) * U256::from((in_size + 31) / 32) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Pricing scheme and execution definition for a built-in contract.
 | ||||||
|  | pub struct Builtin { | ||||||
|  | 	pricer: Box<Pricer>, | ||||||
|  | 	native: Box<Impl>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
| impl Builtin { | impl Builtin { | ||||||
| 	/// Create a new object from components.
 |  | ||||||
| 	pub fn new(cost: Box<Fn(usize) -> U256>, execute: Box<Fn(&[u8], &mut [u8])>) -> Builtin { |  | ||||||
| 		Builtin {cost: cost, execute: execute} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/// Create a new object from a builtin-function name with a linear cost associated with input size.
 |  | ||||||
| 	pub fn from_named_linear(name: &str, base_cost: usize, word_cost: usize) -> Builtin { |  | ||||||
| 		let cost = Box::new(move|s: usize| -> U256 { |  | ||||||
| 			U256::from(base_cost) + U256::from(word_cost) * U256::from((s + 31) / 32) |  | ||||||
| 		}); |  | ||||||
| 
 |  | ||||||
| 		Self::new(cost, new_builtin_exec(name)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/// Simple forwarder for cost.
 | 	/// Simple forwarder for cost.
 | ||||||
| 	pub fn cost(&self, s: usize) -> U256 { (*self.cost)(s) } | 	pub fn cost(&self, s: usize) -> U256 { self.pricer.cost(s) } | ||||||
| 
 | 
 | ||||||
| 	/// Simple forwarder for execute.
 | 	/// Simple forwarder for execute.
 | ||||||
| 	pub fn execute(&self, input: &[u8], output: &mut[u8]) { (*self.execute)(input, output); } | 	pub fn execute(&self, input: &[u8], output: &mut[u8]) { self.native.execute(input, output) } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl From<ethjson::spec::Builtin> for Builtin { | impl From<ethjson::spec::Builtin> for Builtin { | ||||||
| 	fn from(b: ethjson::spec::Builtin) -> Self { | 	fn from(b: ethjson::spec::Builtin) -> Self { | ||||||
| 		match b.pricing { | 		let pricer = match b.pricing { | ||||||
| 			ethjson::spec::Pricing::Linear(linear) => { | 			ethjson::spec::Pricing::Linear(linear) => { | ||||||
| 				Self::from_named_linear(b.name.as_ref(), linear.base, linear.word) | 				Box::new(Linear { | ||||||
|  | 					base: linear.base, | ||||||
|  | 					word: linear.word, | ||||||
|  | 				}) | ||||||
| 			} | 			} | ||||||
|  | 		}; | ||||||
|  | 
 | ||||||
|  | 		Builtin { | ||||||
|  | 			pricer: pricer, | ||||||
|  | 			native: ethereum_builtin(&b.name), | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Copy a bunch of bytes to a destination; if the `src` is too small to fill `dest`,
 | // Ethereum builtin creator.
 | ||||||
| /// leave the rest unchanged.
 | fn ethereum_builtin(name: &str) -> Box<Impl> { | ||||||
| pub fn copy_to(src: &[u8], dest: &mut[u8]) { | 	match name { | ||||||
| 	// NICE: optimise
 | 		"identity" => Box::new(Identity) as Box<Impl>, | ||||||
| 	for i in 0..min(src.len(), dest.len()) { | 		"ecrecover" => Box::new(EcRecover) as Box<Impl>, | ||||||
| 		dest[i] = src[i]; | 		"sha256" => Box::new(Sha256) as Box<Impl>, | ||||||
|  | 		"ripemd160" => Box::new(Ripemd160) as Box<Impl>, | ||||||
|  | 		_ => panic!("invalid builtin name: {}", name), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Create a new builtin executor according to `name`.
 | // Ethereum builtins:
 | ||||||
| /// TODO: turn in to a factory with dynamic registration.
 | //
 | ||||||
| pub fn new_builtin_exec(name: &str) -> Box<Fn(&[u8], &mut [u8])> { | // - The identity function
 | ||||||
| 	match name { | // - ec recovery
 | ||||||
| 		"identity" => Box::new(move|input: &[u8], output: &mut[u8]| { | // - sha256
 | ||||||
| 			for i in 0..min(input.len(), output.len()) { | // - ripemd160
 | ||||||
| 				output[i] = input[i]; | 
 | ||||||
| 			} | #[derive(Debug)] | ||||||
| 		}), | struct Identity; | ||||||
| 		"ecrecover" => Box::new(move|input: &[u8], output: &mut[u8]| { | 
 | ||||||
| 			#[repr(packed)] | #[derive(Debug)] | ||||||
| 			#[derive(Debug, Default)] | struct EcRecover; | ||||||
| 			struct InType { | 
 | ||||||
| 				hash: H256, | #[derive(Debug)] | ||||||
| 				v: H256, | struct Sha256; | ||||||
| 				r: H256, | 
 | ||||||
| 				s: H256, | #[derive(Debug)] | ||||||
| 			} | struct Ripemd160; | ||||||
| 			let mut it = InType::default(); | 
 | ||||||
| 			it.copy_raw(input); | impl Impl for Identity { | ||||||
| 			if it.v == H256::from(&U256::from(27)) || it.v == H256::from(&U256::from(28)) { | 	fn execute(&self, input: &[u8], output: &mut [u8]) { | ||||||
| 				let s = Signature::from_rsv(&it.r, &it.s, it.v[31] - 27); | 		let len = min(input.len(), output.len()); | ||||||
| 				if s.is_valid() { | 		output[..len].copy_from_slice(&input[..len]); | ||||||
| 					if let Ok(p) = recover(&s, &it.hash) { | 	} | ||||||
| 						let r = p.as_slice().sha3(); | } | ||||||
| 						// NICE: optimise and separate out into populate-like function
 | 
 | ||||||
| 						for i in 0..min(32, output.len()) { | impl Impl for EcRecover { | ||||||
| 							output[i] = if i < 12 {0} else {r[i]}; | 	fn execute(&self, i: &[u8], output: &mut [u8]) { | ||||||
| 						} | 		let len = min(i.len(), 128); | ||||||
| 					} | 
 | ||||||
|  | 		let mut input = [0; 128]; | ||||||
|  | 		input[..len].copy_from_slice(&i[..len]); | ||||||
|  | 
 | ||||||
|  | 		let hash = H256::from_slice(&input[0..32]); | ||||||
|  | 		let v = H256::from_slice(&input[32..64]); | ||||||
|  | 		let r = H256::from_slice(&input[64..96]); | ||||||
|  | 		let s = H256::from_slice(&input[96..128]); | ||||||
|  | 
 | ||||||
|  | 		let bit = match v[31] { | ||||||
|  | 			27 | 28 if &v.as_slice()[..31] == &[0; 31] => v[31] - 27, | ||||||
|  | 			_ => return, | ||||||
|  | 		}; | ||||||
|  | 
 | ||||||
|  | 		let s = Signature::from_rsv(&r, &s, bit); | ||||||
|  | 		if s.is_valid() { | ||||||
|  | 			if let Ok(p) = ec_recover(&s, &hash) { | ||||||
|  | 				let r = p.as_slice().sha3(); | ||||||
|  | 
 | ||||||
|  | 				let out_len = min(output.len(), 32); | ||||||
|  | 
 | ||||||
|  | 				for x in &mut output[0.. min(12, out_len)] { | ||||||
|  | 					*x = 0; | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				if out_len > 12 { | ||||||
|  | 					output[12..out_len].copy_from_slice(&r[12..out_len]); | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		}), |  | ||||||
| 		"sha256" => Box::new(move|input: &[u8], output: &mut[u8]| { |  | ||||||
| 			let mut sha = Sha256::new(); |  | ||||||
| 			sha.input(input); |  | ||||||
| 			if output.len() >= 32 { |  | ||||||
| 				sha.result(output); |  | ||||||
| 			} else { |  | ||||||
| 				let mut ret = H256::new(); |  | ||||||
| 				sha.result(ret.as_slice_mut()); |  | ||||||
| 				copy_to(&ret, output); |  | ||||||
| 			} |  | ||||||
| 		}), |  | ||||||
| 		"ripemd160" => Box::new(move|input: &[u8], output: &mut[u8]| { |  | ||||||
| 			let mut sha = Ripemd160::new(); |  | ||||||
| 			sha.input(input); |  | ||||||
| 			let mut ret = H256::new(); |  | ||||||
| 			sha.result(&mut ret.as_slice_mut()[12..32]); |  | ||||||
| 			copy_to(&ret, output); |  | ||||||
| 		}), |  | ||||||
| 		_ => { |  | ||||||
| 			panic!("invalid builtin name {}", name); |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | impl Impl for Sha256 { | ||||||
| fn identity() { | 	fn execute(&self, input: &[u8], output: &mut [u8]) { | ||||||
| 	let f = new_builtin_exec("identity"); | 		let out_len = min(output.len(), 32); | ||||||
| 	let i = [0u8, 1, 2, 3]; |  | ||||||
| 
 | 
 | ||||||
| 	let mut o2 = [255u8; 2]; | 		let mut sha = Sha256Digest::new(); | ||||||
| 	f(&i[..], &mut o2[..]); | 		sha.input(input); | ||||||
| 	assert_eq!(i[0..2], o2); |  | ||||||
| 
 | 
 | ||||||
| 	let mut o4 = [255u8; 4]; | 		if out_len == 32 { | ||||||
| 	f(&i[..], &mut o4[..]); | 			sha.result(&mut output[0..32]); | ||||||
| 	assert_eq!(i, o4); | 		} else { | ||||||
|  | 			let mut out = [0; 32]; | ||||||
|  | 			sha.result(&mut out); | ||||||
| 
 | 
 | ||||||
| 	let mut o8 = [255u8; 8]; | 			output.copy_from_slice(&out[..out_len]) | ||||||
| 	f(&i[..], &mut o8[..]); | 		} | ||||||
| 	assert_eq!(i, o8[..4]); | 	} | ||||||
| 	assert_eq!([255u8; 4], o8[4..]); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | impl Impl for Ripemd160 { | ||||||
| fn sha256() { | 	fn execute(&self, input: &[u8], output: &mut [u8]) { | ||||||
| 	use rustc_serialize::hex::FromHex; | 		let out_len = min(output.len(), 32); | ||||||
| 	let f = new_builtin_exec("sha256"); |  | ||||||
| 	let i = [0u8; 0]; |  | ||||||
| 
 | 
 | ||||||
| 	let mut o = [255u8; 32]; | 		let mut sha = Ripemd160Digest::new(); | ||||||
| 	f(&i[..], &mut o[..]); | 		sha.input(input); | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855").unwrap())[..]); |  | ||||||
| 
 | 
 | ||||||
| 	let mut o8 = [255u8; 8]; | 		for x in &mut output[0.. min(12, out_len)] { | ||||||
| 	f(&i[..], &mut o8[..]); | 			*x = 0; | ||||||
| 	assert_eq!(&o8[..], &(FromHex::from_hex("e3b0c44298fc1c14").unwrap())[..]); | 		} | ||||||
| 
 | 
 | ||||||
| 	let mut o34 = [255u8; 34]; | 		if out_len >= 32 { | ||||||
| 	f(&i[..], &mut o34[..]); | 			sha.result(&mut output[12..32]); | ||||||
| 	assert_eq!(&o34[..], &(FromHex::from_hex("e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855ffff").unwrap())[..]); | 		} else if out_len > 12 { | ||||||
|  | 			let mut out = [0; 20]; | ||||||
|  | 			sha.result(&mut out); | ||||||
|  | 
 | ||||||
|  | 			output.copy_from_slice(&out[12..out_len]) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[cfg(test)] | ||||||
| fn ripemd160() { | mod tests { | ||||||
| 	use rustc_serialize::hex::FromHex; | 	use super::{Builtin, Linear, ethereum_builtin, Pricer}; | ||||||
| 	let f = new_builtin_exec("ripemd160"); | 	use ethjson; | ||||||
| 	let i = [0u8; 0]; | 	use util::U256; | ||||||
| 
 | 
 | ||||||
| 	let mut o = [255u8; 32]; | 	#[test] | ||||||
| 	f(&i[..], &mut o[..]); | 	fn identity() { | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("0000000000000000000000009c1185a5c5e9fc54612808977ee8f548b2258d31").unwrap())[..]); | 		let f = ethereum_builtin("identity"); | ||||||
| 
 | 
 | ||||||
| 	let mut o8 = [255u8; 8]; | 		let i = [0u8, 1, 2, 3]; | ||||||
| 	f(&i[..], &mut o8[..]); |  | ||||||
| 	assert_eq!(&o8[..], &(FromHex::from_hex("0000000000000000").unwrap())[..]); |  | ||||||
| 
 | 
 | ||||||
| 	let mut o34 = [255u8; 34]; | 		let mut o2 = [255u8; 2]; | ||||||
| 	f(&i[..], &mut o34[..]); | 		f.execute(&i[..], &mut o2[..]); | ||||||
| 	assert_eq!(&o34[..], &(FromHex::from_hex("0000000000000000000000009c1185a5c5e9fc54612808977ee8f548b2258d31ffff").unwrap())[..]); | 		assert_eq!(i[0..2], o2); | ||||||
| } | 
 | ||||||
| 
 | 		let mut o4 = [255u8; 4]; | ||||||
| #[test] | 		f.execute(&i[..], &mut o4[..]); | ||||||
| fn ecrecover() { | 		assert_eq!(i, o4); | ||||||
| 	use rustc_serialize::hex::FromHex; | 
 | ||||||
| 	/*let k = KeyPair::from_secret(b"test".sha3()).unwrap();
 | 		let mut o8 = [255u8; 8]; | ||||||
| 	let a: Address = From::from(k.public().sha3()); | 		f.execute(&i[..], &mut o8[..]); | ||||||
| 	println!("Address: {}", a); | 		assert_eq!(i, o8[..4]); | ||||||
| 	let m = b"hello world".sha3(); | 		assert_eq!([255u8; 4], o8[4..]); | ||||||
| 	println!("Message: {}", m); | 	} | ||||||
| 	let s = k.sign(&m).unwrap(); | 
 | ||||||
| 	println!("Signed: {}", s);*/ | 	#[test] | ||||||
| 
 | 	fn sha256() { | ||||||
| 	let f = new_builtin_exec("ecrecover"); | 		use rustc_serialize::hex::FromHex; | ||||||
| 	let i = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b650acf9d3f5f0a2c799776a1254355d5f4061762a237396a99a0e0e3fc2bcd6729514a0dacb2e623ac4abd157cb18163ff942280db4d5caad66ddf941ba12e03").unwrap(); | 		let f = ethereum_builtin("sha256"); | ||||||
| 
 | 
 | ||||||
| 	let mut o = [255u8; 32]; | 		let i = [0u8; 0]; | ||||||
| 	f(&i[..], &mut o[..]); | 
 | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("000000000000000000000000c08b5542d177ac6686946920409741463a15dddb").unwrap())[..]); | 		let mut o = [255u8; 32]; | ||||||
| 
 | 		f.execute(&i[..], &mut o[..]); | ||||||
| 	let mut o8 = [255u8; 8]; | 		assert_eq!(&o[..], &(FromHex::from_hex("e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855").unwrap())[..]); | ||||||
| 	f(&i[..], &mut o8[..]); | 
 | ||||||
| 	assert_eq!(&o8[..], &(FromHex::from_hex("0000000000000000").unwrap())[..]); | 		let mut o8 = [255u8; 8]; | ||||||
| 
 | 		f.execute(&i[..], &mut o8[..]); | ||||||
| 	let mut o34 = [255u8; 34]; | 		assert_eq!(&o8[..], &(FromHex::from_hex("e3b0c44298fc1c14").unwrap())[..]); | ||||||
| 	f(&i[..], &mut o34[..]); | 
 | ||||||
| 	assert_eq!(&o34[..], &(FromHex::from_hex("000000000000000000000000c08b5542d177ac6686946920409741463a15dddbffff").unwrap())[..]); | 		let mut o34 = [255u8; 34]; | ||||||
| 
 | 		f.execute(&i[..], &mut o34[..]); | ||||||
| 	let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001a650acf9d3f5f0a2c799776a1254355d5f4061762a237396a99a0e0e3fc2bcd6729514a0dacb2e623ac4abd157cb18163ff942280db4d5caad66ddf941ba12e03").unwrap(); | 		assert_eq!(&o34[..], &(FromHex::from_hex("e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855ffff").unwrap())[..]); | ||||||
| 	let mut o = [255u8; 32]; | 	} | ||||||
| 	f(&i_bad[..], &mut o[..]); | 
 | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | 	#[test] | ||||||
| 
 | 	fn ripemd160() { | ||||||
| 	let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b000000000000000000000000000000000000000000000000000000000000001b0000000000000000000000000000000000000000000000000000000000000000").unwrap(); | 		use rustc_serialize::hex::FromHex; | ||||||
| 	let mut o = [255u8; 32]; | 		let f = ethereum_builtin("ripemd160"); | ||||||
| 	f(&i_bad[..], &mut o[..]); | 
 | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | 		let i = [0u8; 0]; | ||||||
| 
 | 
 | ||||||
| 	let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001b").unwrap(); | 		let mut o = [255u8; 32]; | ||||||
| 	let mut o = [255u8; 32]; | 		f.execute(&i[..], &mut o[..]); | ||||||
| 	f(&i_bad[..], &mut o[..]); | 		assert_eq!(&o[..], &(FromHex::from_hex("0000000000000000000000009c1185a5c5e9fc54612808977ee8f548b2258d31").unwrap())[..]); | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | 
 | ||||||
| 
 | 		let mut o8 = [255u8; 8]; | ||||||
| 	let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001bffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000000000000000000000000001b").unwrap(); | 		f.execute(&i[..], &mut o8[..]); | ||||||
| 	let mut o = [255u8; 32]; | 		assert_eq!(&o8[..], &(FromHex::from_hex("0000000000000000").unwrap())[..]); | ||||||
| 	f(&i_bad[..], &mut o[..]); | 
 | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | 		let mut o34 = [255u8; 34]; | ||||||
| 
 | 		f.execute(&i[..], &mut o34[..]); | ||||||
| 	let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b000000000000000000000000000000000000000000000000000000000000001bffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap(); | 		assert_eq!(&o34[..], &(FromHex::from_hex("0000000000000000000000009c1185a5c5e9fc54612808977ee8f548b2258d31ffff").unwrap())[..]); | ||||||
| 	let mut o = [255u8; 32]; | 	} | ||||||
| 	f(&i_bad[..], &mut o[..]); | 
 | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | 	#[test] | ||||||
| 
 | 	fn ecrecover() { | ||||||
| 	// TODO: Should this (corrupted version of the above) fail rather than returning some address?
 | 		use rustc_serialize::hex::FromHex; | ||||||
| /*	let i_bad = FromHex::from_hex("48173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b650acf9d3f5f0a2c799776a1254355d5f4061762a237396a99a0e0e3fc2bcd6729514a0dacb2e623ac4abd157cb18163ff942280db4d5caad66ddf941ba12e03").unwrap();
 | 		/*let k = KeyPair::from_secret(b"test".sha3()).unwrap();
 | ||||||
| 	let mut o = [255u8; 32]; | 		let a: Address = From::from(k.public().sha3()); | ||||||
| 	f(&i_bad[..], &mut o[..]); | 		println!("Address: {}", a); | ||||||
| 	assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]);*/ | 		let m = b"hello world".sha3(); | ||||||
| } | 		println!("Message: {}", m); | ||||||
| 
 | 		let s = k.sign(&m).unwrap(); | ||||||
| #[test] | 		println!("Signed: {}", s);*/ | ||||||
| #[should_panic] | 
 | ||||||
| fn from_unknown_linear() { | 		let f = ethereum_builtin("ecrecover"); | ||||||
| 	let _ = Builtin::from_named_linear("dw", 10, 20); | 
 | ||||||
| } | 		let i = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b650acf9d3f5f0a2c799776a1254355d5f4061762a237396a99a0e0e3fc2bcd6729514a0dacb2e623ac4abd157cb18163ff942280db4d5caad66ddf941ba12e03").unwrap(); | ||||||
| 
 | 
 | ||||||
| #[test] | 		let mut o = [255u8; 32]; | ||||||
| fn from_named_linear() { | 		f.execute(&i[..], &mut o[..]); | ||||||
| 	let b = Builtin::from_named_linear("identity", 10, 20); | 		assert_eq!(&o[..], &(FromHex::from_hex("000000000000000000000000c08b5542d177ac6686946920409741463a15dddb").unwrap())[..]); | ||||||
| 	assert_eq!((*b.cost)(0), U256::from(10)); | 
 | ||||||
| 	assert_eq!((*b.cost)(1), U256::from(30)); | 		let mut o8 = [255u8; 8]; | ||||||
| 	assert_eq!((*b.cost)(32), U256::from(30)); | 		f.execute(&i[..], &mut o8[..]); | ||||||
| 	assert_eq!((*b.cost)(33), U256::from(50)); | 		assert_eq!(&o8[..], &(FromHex::from_hex("0000000000000000").unwrap())[..]); | ||||||
| 
 | 
 | ||||||
| 	let i = [0u8, 1, 2, 3]; | 		let mut o34 = [255u8; 34]; | ||||||
| 	let mut o = [255u8; 4]; | 		f.execute(&i[..], &mut o34[..]); | ||||||
| 	(*b.execute)(&i[..], &mut o[..]); | 		assert_eq!(&o34[..], &(FromHex::from_hex("000000000000000000000000c08b5542d177ac6686946920409741463a15dddbffff").unwrap())[..]); | ||||||
| 	assert_eq!(i, o); | 
 | ||||||
| } | 		let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001a650acf9d3f5f0a2c799776a1254355d5f4061762a237396a99a0e0e3fc2bcd6729514a0dacb2e623ac4abd157cb18163ff942280db4d5caad66ddf941ba12e03").unwrap(); | ||||||
| 
 | 		let mut o = [255u8; 32]; | ||||||
| #[test] | 		f.execute(&i_bad[..], &mut o[..]); | ||||||
| fn from_json() { | 		assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | ||||||
| 	let b = Builtin::from(ethjson::spec::Builtin { | 
 | ||||||
| 		name: "identity".to_owned(), | 		let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b000000000000000000000000000000000000000000000000000000000000001b0000000000000000000000000000000000000000000000000000000000000000").unwrap(); | ||||||
| 		pricing: ethjson::spec::Pricing::Linear(ethjson::spec::Linear { | 		let mut o = [255u8; 32]; | ||||||
| 			base: 10, | 		f.execute(&i_bad[..], &mut o[..]); | ||||||
| 			word: 20, | 		assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | ||||||
| 		}) | 
 | ||||||
| 	}); | 		let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001b").unwrap(); | ||||||
| 
 | 		let mut o = [255u8; 32]; | ||||||
| 	assert_eq!((*b.cost)(0), U256::from(10)); | 		f.execute(&i_bad[..], &mut o[..]); | ||||||
| 	assert_eq!((*b.cost)(1), U256::from(30)); | 		assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | ||||||
| 	assert_eq!((*b.cost)(32), U256::from(30)); | 
 | ||||||
| 	assert_eq!((*b.cost)(33), U256::from(50)); | 		let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001bffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000000000000000000000000001b").unwrap(); | ||||||
| 
 | 		let mut o = [255u8; 32]; | ||||||
| 	let i = [0u8, 1, 2, 3]; | 		f.execute(&i_bad[..], &mut o[..]); | ||||||
| 	let mut o = [255u8; 4]; | 		assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | ||||||
| 	(*b.execute)(&i[..], &mut o[..]); | 
 | ||||||
| 	assert_eq!(i, o); | 		let i_bad = FromHex::from_hex("47173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b000000000000000000000000000000000000000000000000000000000000001bffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap(); | ||||||
|  | 		let mut o = [255u8; 32]; | ||||||
|  | 		f.execute(&i_bad[..], &mut o[..]); | ||||||
|  | 		assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]); | ||||||
|  | 
 | ||||||
|  | 		// TODO: Should this (corrupted version of the above) fail rather than returning some address?
 | ||||||
|  | 	/*	let i_bad = FromHex::from_hex("48173285a8d7341e5e972fc677286384f802f8ef42a5ec5f03bbfa254cb01fad000000000000000000000000000000000000000000000000000000000000001b650acf9d3f5f0a2c799776a1254355d5f4061762a237396a99a0e0e3fc2bcd6729514a0dacb2e623ac4abd157cb18163ff942280db4d5caad66ddf941ba12e03").unwrap();
 | ||||||
|  | 		let mut o = [255u8; 32]; | ||||||
|  | 		f.execute(&i_bad[..], &mut o[..]); | ||||||
|  | 		assert_eq!(&o[..], &(FromHex::from_hex("ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff").unwrap())[..]);*/ | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	#[should_panic] | ||||||
|  | 	fn from_unknown_linear() { | ||||||
|  | 		let _ = ethereum_builtin("foo"); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn from_named_linear() { | ||||||
|  | 		let pricer = Box::new(Linear { base: 10, word: 20 }); | ||||||
|  | 		let b = Builtin { | ||||||
|  | 			pricer: pricer as Box<Pricer>, | ||||||
|  | 			native: ethereum_builtin("identity"), | ||||||
|  | 		}; | ||||||
|  | 
 | ||||||
|  | 		assert_eq!(b.cost(0), U256::from(10)); | ||||||
|  | 		assert_eq!(b.cost(1), U256::from(30)); | ||||||
|  | 		assert_eq!(b.cost(32), U256::from(30)); | ||||||
|  | 		assert_eq!(b.cost(33), U256::from(50)); | ||||||
|  | 
 | ||||||
|  | 		let i = [0u8, 1, 2, 3]; | ||||||
|  | 		let mut o = [255u8; 4]; | ||||||
|  | 		b.execute(&i[..], &mut o[..]); | ||||||
|  | 		assert_eq!(i, o); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	#[test] | ||||||
|  | 	fn from_json() { | ||||||
|  | 		let b = Builtin::from(ethjson::spec::Builtin { | ||||||
|  | 			name: "identity".to_owned(), | ||||||
|  | 			pricing: ethjson::spec::Pricing::Linear(ethjson::spec::Linear { | ||||||
|  | 				base: 10, | ||||||
|  | 				word: 20, | ||||||
|  | 			}) | ||||||
|  | 		}); | ||||||
|  | 
 | ||||||
|  | 		assert_eq!(b.cost(0), U256::from(10)); | ||||||
|  | 		assert_eq!(b.cost(1), U256::from(30)); | ||||||
|  | 		assert_eq!(b.cost(32), U256::from(30)); | ||||||
|  | 		assert_eq!(b.cost(33), U256::from(50)); | ||||||
|  | 
 | ||||||
|  | 		let i = [0u8, 1, 2, 3]; | ||||||
|  | 		let mut o = [255u8; 4]; | ||||||
|  | 		b.execute(&i[..], &mut o[..]); | ||||||
|  | 		assert_eq!(i, o); | ||||||
|  | 	} | ||||||
| } | } | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user