2020-09-22 14:53:52 +02:00
|
|
|
// Copyright 2015-2020 Parity Technologies (UK) Ltd.
|
|
|
|
// This file is part of OpenEthereum.
|
2016-07-11 09:46:33 +02:00
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
// OpenEthereum is free software: you can redistribute it and/or modify
|
2016-07-11 09:46:33 +02:00
|
|
|
// 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.
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
// OpenEthereum is distributed in the hope that it will be useful,
|
2016-07-11 09:46:33 +02:00
|
|
|
// 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
|
2020-09-22 14:53:52 +02:00
|
|
|
// along with OpenEthereum. If not, see <http://www.gnu.org/licenses/>.
|
2016-07-11 09:46:33 +02:00
|
|
|
|
2016-02-05 01:49:17 +01:00
|
|
|
//! DB backend wrapper for Account trie
|
2018-01-10 13:35:18 +01:00
|
|
|
use ethereum_types::H256;
|
2020-08-05 06:08:03 +02:00
|
|
|
use hash::{keccak, KECCAK_NULL_RLP};
|
|
|
|
use hash_db::{AsHashDB, HashDB};
|
2018-07-02 18:50:05 +02:00
|
|
|
use keccak_hasher::KeccakHasher;
|
2018-01-10 13:35:18 +01:00
|
|
|
use kvdb::DBValue;
|
2016-09-01 14:29:59 +02:00
|
|
|
use rlp::NULL_RLP;
|
2016-02-05 01:49:17 +01:00
|
|
|
|
2018-01-10 13:35:18 +01:00
|
|
|
#[cfg(test)]
|
|
|
|
use ethereum_types::Address;
|
|
|
|
|
2016-07-11 09:46:33 +02:00
|
|
|
// combines a key with an address hash to ensure uniqueness.
|
|
|
|
// leaves the first 96 bits untouched in order to support partial key lookup.
|
|
|
|
#[inline]
|
|
|
|
fn combine_key<'a>(address_hash: &'a H256, key: &'a H256) -> H256 {
|
2020-08-05 06:08:03 +02:00
|
|
|
let mut dst = key.clone();
|
|
|
|
{
|
|
|
|
let last_src: &[u8] = &*address_hash;
|
|
|
|
let last_dst: &mut [u8] = &mut *dst;
|
|
|
|
for (k, a) in last_dst[12..].iter_mut().zip(&last_src[12..]) {
|
|
|
|
*k ^= *a
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dst
|
2016-07-11 09:46:33 +02:00
|
|
|
}
|
|
|
|
|
2016-08-24 16:53:36 +02:00
|
|
|
/// A factory for different kinds of account dbs.
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub enum Factory {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Mangle hashes based on address. This is the default.
|
|
|
|
Mangled,
|
|
|
|
/// Don't mangle hashes.
|
|
|
|
Plain,
|
2016-08-24 16:53:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Factory {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn default() -> Self {
|
|
|
|
Factory::Mangled
|
|
|
|
}
|
2016-08-24 16:53:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Factory {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Create a read-only accountdb.
|
|
|
|
/// This will panic when write operations are called.
|
|
|
|
pub fn readonly<'db>(
|
|
|
|
&self,
|
2020-07-29 10:36:15 +02:00
|
|
|
db: &'db dyn HashDB<KeccakHasher, DBValue>,
|
2020-08-05 06:08:03 +02:00
|
|
|
address_hash: H256,
|
2020-07-29 10:36:15 +02:00
|
|
|
) -> Box<dyn HashDB<KeccakHasher, DBValue> + 'db> {
|
2020-08-05 06:08:03 +02:00
|
|
|
match *self {
|
|
|
|
Factory::Mangled => Box::new(AccountDB::from_hash(db, address_hash)),
|
|
|
|
Factory::Plain => Box::new(Wrapping(db)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a new mutable hashdb.
|
|
|
|
pub fn create<'db>(
|
|
|
|
&self,
|
2020-07-29 10:36:15 +02:00
|
|
|
db: &'db mut dyn HashDB<KeccakHasher, DBValue>,
|
2020-08-05 06:08:03 +02:00
|
|
|
address_hash: H256,
|
2020-07-29 10:36:15 +02:00
|
|
|
) -> Box<dyn HashDB<KeccakHasher, DBValue> + 'db> {
|
2020-08-05 06:08:03 +02:00
|
|
|
match *self {
|
|
|
|
Factory::Mangled => Box::new(AccountDBMut::from_hash(db, address_hash)),
|
|
|
|
Factory::Plain => Box::new(WrappingMut(db)),
|
|
|
|
}
|
|
|
|
}
|
2016-08-24 16:53:36 +02:00
|
|
|
}
|
|
|
|
|
2016-02-05 01:49:17 +01:00
|
|
|
// TODO: introduce HashDBMut?
|
|
|
|
/// DB backend wrapper for Account trie
|
|
|
|
/// Transforms trie node keys for the database
|
|
|
|
pub struct AccountDB<'db> {
|
2020-07-29 10:36:15 +02:00
|
|
|
db: &'db dyn HashDB<KeccakHasher, DBValue>,
|
2020-08-05 06:08:03 +02:00
|
|
|
address_hash: H256,
|
2016-02-05 01:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'db> AccountDB<'db> {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Create a new AccountDB from an address.
|
|
|
|
#[cfg(test)]
|
2020-07-29 10:36:15 +02:00
|
|
|
pub fn new(db: &'db dyn HashDB<KeccakHasher, DBValue>, address: &Address) -> Self {
|
2020-08-05 06:08:03 +02:00
|
|
|
Self::from_hash(db, keccak(address))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a new AcountDB from an address' hash.
|
2020-07-29 10:36:15 +02:00
|
|
|
pub fn from_hash(db: &'db dyn HashDB<KeccakHasher, DBValue>, address_hash: H256) -> Self {
|
2020-08-05 06:08:03 +02:00
|
|
|
AccountDB {
|
|
|
|
db: db,
|
|
|
|
address_hash: address_hash,
|
|
|
|
}
|
|
|
|
}
|
2016-02-05 01:49:17 +01:00
|
|
|
}
|
|
|
|
|
2018-10-09 22:07:25 +02:00
|
|
|
impl<'db> AsHashDB<KeccakHasher, DBValue> for AccountDB<'db> {
|
2020-07-29 10:36:15 +02:00
|
|
|
fn as_hash_db(&self) -> &dyn HashDB<KeccakHasher, DBValue> {
|
2020-08-05 06:08:03 +02:00
|
|
|
self
|
|
|
|
}
|
2020-07-29 10:36:15 +02:00
|
|
|
fn as_hash_db_mut(&mut self) -> &mut dyn HashDB<KeccakHasher, DBValue> {
|
2020-08-05 06:08:03 +02:00
|
|
|
self
|
|
|
|
}
|
2018-07-02 18:50:05 +02:00
|
|
|
}
|
|
|
|
|
2018-10-09 22:07:25 +02:00
|
|
|
impl<'db> HashDB<KeccakHasher, DBValue> for AccountDB<'db> {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn get(&self, key: &H256) -> Option<DBValue> {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return Some(DBValue::from_slice(&NULL_RLP));
|
|
|
|
}
|
|
|
|
self.db.get(&combine_key(&self.address_hash, key))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn contains(&self, key: &H256) -> bool {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
self.db.contains(&combine_key(&self.address_hash, key))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn insert(&mut self, _value: &[u8]) -> H256 {
|
|
|
|
unimplemented!()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn emplace(&mut self, _key: H256, _value: DBValue) {
|
|
|
|
unimplemented!()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn remove(&mut self, _key: &H256) {
|
|
|
|
unimplemented!()
|
|
|
|
}
|
2016-02-05 01:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// DB backend wrapper for Account trie
|
|
|
|
pub struct AccountDBMut<'db> {
|
2020-07-29 10:36:15 +02:00
|
|
|
db: &'db mut dyn HashDB<KeccakHasher, DBValue>,
|
2020-08-05 06:08:03 +02:00
|
|
|
address_hash: H256,
|
2016-02-05 01:49:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'db> AccountDBMut<'db> {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Create a new AccountDB from an address.
|
|
|
|
#[cfg(test)]
|
2020-07-29 10:36:15 +02:00
|
|
|
pub fn new(db: &'db mut dyn HashDB<KeccakHasher, DBValue>, address: &Address) -> Self {
|
2020-08-05 06:08:03 +02:00
|
|
|
Self::from_hash(db, keccak(address))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a new AcountDB from an address' hash.
|
2020-07-29 10:36:15 +02:00
|
|
|
pub fn from_hash(db: &'db mut dyn HashDB<KeccakHasher, DBValue>, address_hash: H256) -> Self {
|
2020-08-05 06:08:03 +02:00
|
|
|
AccountDBMut {
|
|
|
|
db: db,
|
|
|
|
address_hash: address_hash,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
pub fn immutable(&'db self) -> AccountDB<'db> {
|
|
|
|
AccountDB {
|
|
|
|
db: self.db,
|
|
|
|
address_hash: self.address_hash.clone(),
|
|
|
|
}
|
|
|
|
}
|
2016-02-05 01:49:17 +01:00
|
|
|
}
|
|
|
|
|
2020-08-05 06:08:03 +02:00
|
|
|
impl<'db> HashDB<KeccakHasher, DBValue> for AccountDBMut<'db> {
|
|
|
|
fn get(&self, key: &H256) -> Option<DBValue> {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return Some(DBValue::from_slice(&NULL_RLP));
|
|
|
|
}
|
|
|
|
self.db.get(&combine_key(&self.address_hash, key))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn contains(&self, key: &H256) -> bool {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
self.db.contains(&combine_key(&self.address_hash, key))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn insert(&mut self, value: &[u8]) -> H256 {
|
|
|
|
if value == &NULL_RLP {
|
|
|
|
return KECCAK_NULL_RLP.clone();
|
|
|
|
}
|
|
|
|
let k = keccak(value);
|
|
|
|
let ak = combine_key(&self.address_hash, &k);
|
|
|
|
self.db.emplace(ak, DBValue::from_slice(value));
|
|
|
|
k
|
|
|
|
}
|
|
|
|
|
|
|
|
fn emplace(&mut self, key: H256, value: DBValue) {
|
|
|
|
if key == KECCAK_NULL_RLP {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let key = combine_key(&self.address_hash, &key);
|
|
|
|
self.db.emplace(key, value)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn remove(&mut self, key: &H256) {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let key = combine_key(&self.address_hash, key);
|
|
|
|
self.db.remove(&key)
|
|
|
|
}
|
2016-02-05 01:49:17 +01:00
|
|
|
}
|
|
|
|
|
2018-10-09 22:07:25 +02:00
|
|
|
impl<'db> AsHashDB<KeccakHasher, DBValue> for AccountDBMut<'db> {
|
2020-07-29 10:36:15 +02:00
|
|
|
fn as_hash_db(&self) -> &dyn HashDB<KeccakHasher, DBValue> {
|
2020-08-05 06:08:03 +02:00
|
|
|
self
|
|
|
|
}
|
2020-07-29 10:36:15 +02:00
|
|
|
fn as_hash_db_mut(&mut self) -> &mut dyn HashDB<KeccakHasher, DBValue> {
|
2020-08-05 06:08:03 +02:00
|
|
|
self
|
|
|
|
}
|
2018-07-02 18:50:05 +02:00
|
|
|
}
|
|
|
|
|
2020-07-29 10:36:15 +02:00
|
|
|
struct Wrapping<'db>(&'db dyn HashDB<KeccakHasher, DBValue>);
|
2016-08-24 16:53:36 +02:00
|
|
|
|
2018-10-09 22:07:25 +02:00
|
|
|
impl<'db> AsHashDB<KeccakHasher, DBValue> for Wrapping<'db> {
|
2020-07-29 10:36:15 +02:00
|
|
|
fn as_hash_db(&self) -> &dyn HashDB<KeccakHasher, DBValue> {
|
2020-08-05 06:08:03 +02:00
|
|
|
self
|
|
|
|
}
|
2020-07-29 10:36:15 +02:00
|
|
|
fn as_hash_db_mut(&mut self) -> &mut dyn HashDB<KeccakHasher, DBValue> {
|
2020-08-05 06:08:03 +02:00
|
|
|
self
|
|
|
|
}
|
2018-07-02 18:50:05 +02:00
|
|
|
}
|
|
|
|
|
2018-10-09 22:07:25 +02:00
|
|
|
impl<'db> HashDB<KeccakHasher, DBValue> for Wrapping<'db> {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn get(&self, key: &H256) -> Option<DBValue> {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return Some(DBValue::from_slice(&NULL_RLP));
|
|
|
|
}
|
|
|
|
self.0.get(key)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn contains(&self, key: &H256) -> bool {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
self.0.contains(key)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn insert(&mut self, _value: &[u8]) -> H256 {
|
|
|
|
unimplemented!()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn emplace(&mut self, _key: H256, _value: DBValue) {
|
|
|
|
unimplemented!()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn remove(&mut self, _key: &H256) {
|
|
|
|
unimplemented!()
|
|
|
|
}
|
2016-08-24 16:53:36 +02:00
|
|
|
}
|
|
|
|
|
2020-07-29 10:36:15 +02:00
|
|
|
struct WrappingMut<'db>(&'db mut dyn HashDB<KeccakHasher, DBValue>);
|
2018-10-09 22:07:25 +02:00
|
|
|
impl<'db> AsHashDB<KeccakHasher, DBValue> for WrappingMut<'db> {
|
2020-07-29 10:36:15 +02:00
|
|
|
fn as_hash_db(&self) -> &dyn HashDB<KeccakHasher, DBValue> {
|
2020-08-05 06:08:03 +02:00
|
|
|
self
|
|
|
|
}
|
2020-07-29 10:36:15 +02:00
|
|
|
fn as_hash_db_mut(&mut self) -> &mut dyn HashDB<KeccakHasher, DBValue> {
|
2020-08-05 06:08:03 +02:00
|
|
|
self
|
|
|
|
}
|
2018-07-02 18:50:05 +02:00
|
|
|
}
|
2016-08-24 16:53:36 +02:00
|
|
|
|
2020-08-05 06:08:03 +02:00
|
|
|
impl<'db> HashDB<KeccakHasher, DBValue> for WrappingMut<'db> {
|
|
|
|
fn get(&self, key: &H256) -> Option<DBValue> {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return Some(DBValue::from_slice(&NULL_RLP));
|
|
|
|
}
|
|
|
|
self.0.get(key)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn contains(&self, key: &H256) -> bool {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
self.0.contains(key)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn insert(&mut self, value: &[u8]) -> H256 {
|
|
|
|
if value == &NULL_RLP {
|
|
|
|
return KECCAK_NULL_RLP.clone();
|
|
|
|
}
|
|
|
|
self.0.insert(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn emplace(&mut self, key: H256, value: DBValue) {
|
|
|
|
if key == KECCAK_NULL_RLP {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.0.emplace(key, value)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn remove(&mut self, key: &H256) {
|
|
|
|
if key == &KECCAK_NULL_RLP {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.0.remove(key)
|
|
|
|
}
|
2016-10-26 13:53:47 +02:00
|
|
|
}
|