2015-11-25 19:26:40 +01:00
|
|
|
use std::str::FromStr;
|
2015-11-26 00:02:43 +01:00
|
|
|
use std::fmt;
|
2015-11-26 23:24:01 +01:00
|
|
|
use std::hash::{Hash, Hasher};
|
2015-11-27 23:24:01 +01:00
|
|
|
use std::ops::{Index, IndexMut, BitOr};
|
2015-11-26 23:24:01 +01:00
|
|
|
use rustc_serialize::hex::*;
|
|
|
|
use error::EthcoreError;
|
|
|
|
use rand::Rng;
|
|
|
|
use rand::os::OsRng;
|
2015-11-27 17:54:33 +01:00
|
|
|
use bytes::BytesConvertable;
|
2015-11-25 19:26:40 +01:00
|
|
|
|
2015-11-27 20:10:31 +01:00
|
|
|
/// types implementing FixedHash must be also BytesConvertable
|
2015-11-28 01:38:00 +01:00
|
|
|
pub trait FixedHash: Sized + BytesConvertable {
|
|
|
|
fn new() -> Self;
|
2015-11-27 20:10:31 +01:00
|
|
|
fn random() -> Self;
|
|
|
|
fn randomize(&mut self);
|
|
|
|
fn mut_bytes(&mut self) -> &mut [u8];
|
2015-11-28 01:38:00 +01:00
|
|
|
fn shift_bloom<'a, T>(&'a mut self, b: &T) -> &'a mut Self where T: FixedHash;
|
|
|
|
fn bloom_part<T>(&self) -> T where T: FixedHash;
|
2015-11-27 20:10:31 +01:00
|
|
|
}
|
|
|
|
|
2015-11-25 19:26:40 +01:00
|
|
|
macro_rules! impl_hash {
|
|
|
|
($from: ident, $size: expr) => {
|
2015-11-27 00:30:21 +01:00
|
|
|
#[derive(Eq)]
|
2015-11-26 00:02:43 +01:00
|
|
|
pub struct $from (pub [u8; $size]);
|
2015-11-25 19:26:40 +01:00
|
|
|
|
2015-11-27 20:10:31 +01:00
|
|
|
impl BytesConvertable for $from {
|
|
|
|
fn bytes(&self) -> &[u8] {
|
|
|
|
&self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FixedHash for $from {
|
2015-11-28 01:38:00 +01:00
|
|
|
fn new() -> $from {
|
|
|
|
$from([0; $size])
|
|
|
|
}
|
|
|
|
|
2015-11-27 20:10:31 +01:00
|
|
|
fn random() -> $from {
|
2015-11-26 23:24:01 +01:00
|
|
|
let mut hash = $from::new();
|
|
|
|
hash.randomize();
|
|
|
|
hash
|
|
|
|
}
|
2015-11-27 20:10:31 +01:00
|
|
|
|
|
|
|
fn randomize(&mut self) {
|
2015-11-26 23:24:01 +01:00
|
|
|
let mut rng = OsRng::new().unwrap();
|
|
|
|
rng.fill_bytes(&mut self.0);
|
|
|
|
}
|
2015-11-27 17:54:33 +01:00
|
|
|
|
2015-11-27 20:10:31 +01:00
|
|
|
fn mut_bytes(&mut self) -> &mut [u8] {
|
2015-11-27 17:54:33 +01:00
|
|
|
&mut self.0
|
|
|
|
}
|
2015-11-28 01:38:00 +01:00
|
|
|
|
|
|
|
fn shift_bloom<'a, T>(&'a mut self, b: &T) -> &'a mut Self where T: FixedHash {
|
|
|
|
let bp: Self = b.bloom_part();
|
|
|
|
let new_self = &bp | self;
|
|
|
|
|
|
|
|
// impl |= instead
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
use std::{mem, ptr};
|
|
|
|
ptr::copy(new_self.0.as_ptr(), self.0.as_mut_ptr(), mem::size_of::<Self>());
|
|
|
|
}
|
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn bloom_part<T>(&self) -> T where T: FixedHash {
|
|
|
|
panic!()
|
|
|
|
}
|
2015-11-27 17:54:33 +01:00
|
|
|
}
|
|
|
|
|
2015-11-25 19:26:40 +01:00
|
|
|
impl FromStr for $from {
|
|
|
|
type Err = EthcoreError;
|
|
|
|
|
|
|
|
fn from_str(s: &str) -> Result<$from, EthcoreError> {
|
|
|
|
let a = try!(s.from_hex());
|
|
|
|
if a.len() != $size { return Err(EthcoreError::BadSize); }
|
|
|
|
let mut ret = $from([0;$size]);
|
|
|
|
for i in 0..$size {
|
|
|
|
ret.0[i] = a[i];
|
|
|
|
}
|
|
|
|
Ok(ret)
|
|
|
|
}
|
|
|
|
}
|
2015-11-26 00:02:43 +01:00
|
|
|
|
|
|
|
impl fmt::Debug for $from {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
for i in self.0.iter() {
|
|
|
|
try!(write!(f, "{:02x}", i));
|
|
|
|
}
|
|
|
|
Ok(())
|
2015-11-26 23:24:01 +01:00
|
|
|
}
|
|
|
|
}
|
2015-11-26 00:02:43 +01:00
|
|
|
impl fmt::Display for $from {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
(self as &fmt::Debug).fmt(f)
|
2015-11-26 23:24:01 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Clone for $from {
|
|
|
|
fn clone(&self) -> $from {
|
2015-11-27 23:24:01 +01:00
|
|
|
unsafe {
|
|
|
|
use std::{mem, ptr};
|
|
|
|
let mut ret: $from = mem::uninitialized();
|
|
|
|
ptr::copy(self.0.as_ptr(), ret.0.as_mut_ptr(), mem::size_of::<$from>());
|
|
|
|
ret
|
|
|
|
}
|
2015-11-26 23:24:01 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-26 00:02:43 +01:00
|
|
|
impl PartialEq for $from {
|
2015-11-26 23:24:01 +01:00
|
|
|
fn eq(&self, other: &Self) -> bool {
|
2015-11-26 00:02:43 +01:00
|
|
|
for i in 0..$size {
|
2015-11-26 23:24:01 +01:00
|
|
|
if self.0[i] != other.0[i] {
|
|
|
|
return false;
|
|
|
|
}
|
2015-11-26 00:02:43 +01:00
|
|
|
}
|
|
|
|
true
|
2015-11-26 23:24:01 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Hash for $from {
|
|
|
|
fn hash<H>(&self, state: &mut H) where H: Hasher {
|
|
|
|
state.write(&self.0);
|
|
|
|
state.finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Index<usize> for $from {
|
|
|
|
type Output = u8;
|
|
|
|
|
|
|
|
fn index<'a>(&'a self, index: usize) -> &'a u8 {
|
|
|
|
&self.0[index]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl IndexMut<usize> for $from {
|
|
|
|
fn index_mut<'a>(&'a mut self, index: usize) -> &'a mut u8 {
|
|
|
|
&mut self.0[index]
|
|
|
|
}
|
2015-11-26 00:02:43 +01:00
|
|
|
}
|
2015-11-27 23:24:01 +01:00
|
|
|
|
|
|
|
impl<'a> BitOr for &'a $from {
|
|
|
|
type Output = $from;
|
|
|
|
|
|
|
|
fn bitor(self, rhs: Self) -> Self::Output {
|
|
|
|
unsafe {
|
|
|
|
use std::mem;
|
|
|
|
let mut ret: $from = mem::uninitialized();
|
|
|
|
for i in 0..$size {
|
|
|
|
ret.0[i] = self.0[i] | rhs.0[i];
|
|
|
|
}
|
|
|
|
ret
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl BitOr for $from {
|
|
|
|
type Output = $from;
|
|
|
|
|
|
|
|
fn bitor(self, rhs: Self) -> Self::Output {
|
|
|
|
&self | &rhs
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-25 19:26:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-27 13:16:32 +01:00
|
|
|
impl_hash!(H64, 8);
|
|
|
|
impl_hash!(H128, 16);
|
2015-11-25 19:26:40 +01:00
|
|
|
impl_hash!(Address, 20);
|
2015-11-27 13:16:32 +01:00
|
|
|
impl_hash!(H256, 32);
|
|
|
|
impl_hash!(H512, 64);
|
|
|
|
impl_hash!(H520, 65);
|
|
|
|
impl_hash!(H1024, 128);
|
|
|
|
impl_hash!(H2048, 256);
|
2015-11-25 19:26:40 +01:00
|
|
|
|
|
|
|
#[test]
|
2015-11-26 21:34:28 +01:00
|
|
|
fn hash() {
|
2015-11-27 13:16:32 +01:00
|
|
|
let h = H64([0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef]);
|
|
|
|
assert_eq!(H64::from_str("0123456789abcdef").unwrap(), h);
|
2015-11-26 00:27:20 +01:00
|
|
|
assert_eq!(format!("{}", h), "0123456789abcdef");
|
|
|
|
assert_eq!(format!("{:?}", h), "0123456789abcdef");
|
|
|
|
assert!(h == h);
|
2015-11-27 13:16:32 +01:00
|
|
|
assert!(h != H64([0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xee]));
|
|
|
|
assert!(h != H64([0; 8]));
|
2015-11-26 23:24:01 +01:00
|
|
|
}
|
2015-11-27 23:24:01 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn hash_bitor() {
|
|
|
|
let a = H64([1; 8]);
|
|
|
|
let b = H64([2; 8]);
|
|
|
|
let c = H64([3; 8]);
|
|
|
|
|
|
|
|
// borrow
|
|
|
|
assert_eq!(&a | &b, c);
|
|
|
|
|
|
|
|
// move
|
|
|
|
assert_eq!(a | b, c);
|
|
|
|
}
|