9e719f088f
* Add license and readme * Use pure rust implementation * Bump version to 0.1.1 * Don't use C, prefer the pure Rust implementation * Add test for `write_keccak` * Bump version * Add benchmarks * Add benchmarks * Add keccak_256, keccak_512, keccak_256_unchecked and keccak_512_unchecked – mostly for compatibility with ethash * Remove failed git merge attempt from external git repo Cargo.lock updates * whitespace * Mark unsafe function unsafe * Unsafe calls in unsafe block * Document unsafety invariants * Revert unintended changes to Cargo.lock
143 lines
4.7 KiB
Rust
143 lines
4.7 KiB
Rust
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
|
// This file is part of Parity.
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
extern crate ethereum_types;
|
|
extern crate tiny_keccak;
|
|
|
|
use std::io;
|
|
use std::slice;
|
|
use tiny_keccak::Keccak;
|
|
|
|
pub use ethereum_types::H256;
|
|
|
|
/// Get the KECCAK (i.e. Keccak) hash of the empty bytes string.
|
|
pub const KECCAK_EMPTY: H256 = H256( [0xc5, 0xd2, 0x46, 0x01, 0x86, 0xf7, 0x23, 0x3c, 0x92, 0x7e, 0x7d, 0xb2, 0xdc, 0xc7, 0x03, 0xc0, 0xe5, 0x00, 0xb6, 0x53, 0xca, 0x82, 0x27, 0x3b, 0x7b, 0xfa, 0xd8, 0x04, 0x5d, 0x85, 0xa4, 0x70] );
|
|
|
|
/// The KECCAK of the RLP encoding of empty data.
|
|
pub const KECCAK_NULL_RLP: H256 = H256( [0x56, 0xe8, 0x1f, 0x17, 0x1b, 0xcc, 0x55, 0xa6, 0xff, 0x83, 0x45, 0xe6, 0x92, 0xc0, 0xf8, 0x6e, 0x5b, 0x48, 0xe0, 0x1b, 0x99, 0x6c, 0xad, 0xc0, 0x01, 0x62, 0x2f, 0xb5, 0xe3, 0x63, 0xb4, 0x21] );
|
|
|
|
/// The KECCAK of the RLP encoding of empty list.
|
|
pub const KECCAK_EMPTY_LIST_RLP: H256 = H256( [0x1d, 0xcc, 0x4d, 0xe8, 0xde, 0xc7, 0x5d, 0x7a, 0xab, 0x85, 0xb5, 0x67, 0xb6, 0xcc, 0xd4, 0x1a, 0xd3, 0x12, 0x45, 0x1b, 0x94, 0x8a, 0x74, 0x13, 0xf0, 0xa1, 0x42, 0xfd, 0x40, 0xd4, 0x93, 0x47] );
|
|
|
|
|
|
pub fn keccak<T: AsRef<[u8]>>(s: T) -> H256 {
|
|
let mut result = [0u8; 32];
|
|
write_keccak(s, &mut result);
|
|
H256(result)
|
|
}
|
|
|
|
pub unsafe fn keccak_256_unchecked(out: *mut u8, outlen: usize, input: *const u8, inputlen: usize) {
|
|
// This is safe since `keccak_*` uses an internal buffer and copies the result to the output. This
|
|
// means that we can reuse the input buffer for both input and output.
|
|
Keccak::keccak256(
|
|
slice::from_raw_parts(input, inputlen),
|
|
slice::from_raw_parts_mut(out, outlen)
|
|
);
|
|
}
|
|
|
|
pub unsafe fn keccak_512_unchecked(out: *mut u8, outlen: usize, input: *const u8, inputlen: usize) {
|
|
// This is safe since `keccak_*` uses an internal buffer and copies the result to the output. This
|
|
// means that we can reuse the input buffer for both input and output.
|
|
Keccak::keccak512(
|
|
slice::from_raw_parts(input, inputlen),
|
|
slice::from_raw_parts_mut(out, outlen)
|
|
);
|
|
}
|
|
|
|
pub fn keccak_256(input: &[u8], mut output: &mut [u8]) { Keccak::keccak256(input, &mut output); }
|
|
|
|
pub fn keccak_512(input: &[u8], mut output: &mut [u8]) { Keccak::keccak512(input, &mut output); }
|
|
|
|
pub fn write_keccak<T: AsRef<[u8]>>(s: T, dest: &mut [u8]) { Keccak::keccak256(s.as_ref(), dest); }
|
|
|
|
pub fn keccak_pipe(r: &mut io::BufRead, w: &mut io::Write) -> Result<H256, io::Error> {
|
|
let mut output = [0u8; 32];
|
|
let mut input = [0u8; 1024];
|
|
let mut keccak = Keccak::new_keccak256();
|
|
|
|
// read file
|
|
loop {
|
|
let some = r.read(&mut input)?;
|
|
if some == 0 {
|
|
break;
|
|
}
|
|
keccak.update(&input[0..some]);
|
|
w.write_all(&input[0..some])?;
|
|
}
|
|
|
|
keccak.finalize(&mut output);
|
|
Ok(output.into())
|
|
}
|
|
|
|
pub fn keccak_buffer(r: &mut io::BufRead) -> Result<H256, io::Error> {
|
|
keccak_pipe(r, &mut io::sink())
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
extern crate tempdir;
|
|
|
|
use std::fs;
|
|
use std::io::{Write, BufReader};
|
|
use self::tempdir::TempDir;
|
|
use super::{keccak, write_keccak, keccak_buffer, KECCAK_EMPTY};
|
|
|
|
#[test]
|
|
fn keccak_empty() {
|
|
assert_eq!(keccak([0u8; 0]), KECCAK_EMPTY);
|
|
}
|
|
|
|
#[test]
|
|
fn keccak_as() {
|
|
assert_eq!(keccak([0x41u8; 32]), From::from("59cad5948673622c1d64e2322488bf01619f7ff45789741b15a9f782ce9290a8"));
|
|
}
|
|
|
|
#[test]
|
|
fn write_keccak_with_content() {
|
|
let data: Vec<u8> = From::from("hello world");
|
|
let expected = vec![
|
|
0x47, 0x17, 0x32, 0x85, 0xa8, 0xd7, 0x34, 0x1e,
|
|
0x5e, 0x97, 0x2f, 0xc6, 0x77, 0x28, 0x63, 0x84,
|
|
0xf8, 0x02, 0xf8, 0xef, 0x42, 0xa5, 0xec, 0x5f,
|
|
0x03, 0xbb, 0xfa, 0x25, 0x4c, 0xb0, 0x1f, 0xad
|
|
];
|
|
let mut dest = [0u8;32];
|
|
write_keccak(data, &mut dest);
|
|
|
|
assert_eq!(dest, expected.as_ref());
|
|
}
|
|
|
|
#[test]
|
|
fn should_keccak_a_file() {
|
|
// given
|
|
let tempdir = TempDir::new("keccak").unwrap();
|
|
let mut path = tempdir.path().to_owned();
|
|
path.push("should_keccak_a_file");
|
|
// Prepare file
|
|
{
|
|
let mut file = fs::File::create(&path).unwrap();
|
|
file.write_all(b"something").unwrap();
|
|
}
|
|
|
|
let mut file = BufReader::new(fs::File::open(&path).unwrap());
|
|
// when
|
|
let hash = keccak_buffer(&mut file).unwrap();
|
|
|
|
// then
|
|
assert_eq!(format!("{:x}", hash), "68371d7e884c168ae2022c82bd837d51837718a7f7dfb7aa3f753074a35e1d87");
|
|
}
|
|
}
|