| 
									
										
										
										
											2019-11-18 16:47:01 -08:00
										 |  |  | use bytecode_verifier::verifier::VerifiedModule;
 | 
					
						
							| 
									
										
										
										
											2019-10-29 10:39:10 -07:00
										 |  |  | use canonical_serialization::SimpleDeserializer;
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  | use failure::prelude::*;
 | 
					
						
							| 
									
										
										
										
											2019-07-24 21:43:14 -07:00
										 |  |  | use indexmap::IndexMap;
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  | use log::*;
 | 
					
						
							|  |  |  | use state_view::StateView;
 | 
					
						
							|  |  |  | use types::{
 | 
					
						
							|  |  |  |     access_path::AccessPath,
 | 
					
						
							|  |  |  |     account_address::AccountAddress,
 | 
					
						
							| 
									
										
										
										
											2019-10-29 10:39:10 -07:00
										 |  |  |     account_config::{self, AccountResource},
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |     write_set::{WriteOp, WriteSet, WriteSetMut},
 | 
					
						
							|  |  |  | };
 | 
					
						
							| 
									
										
										
										
											2019-11-18 16:47:01 -08:00
										 |  |  | use vm::access::ModuleAccess;
 | 
					
						
							|  |  |  | use vm::errors::VMResult;
 | 
					
						
							| 
									
										
										
										
											2019-10-29 10:39:10 -07:00
										 |  |  | use vm_runtime::{data_cache::RemoteCache, identifier::create_access_path};
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | /// An in-memory implementation of [`StateView`] and [`RemoteCache`] for the VM.
 | 
					
						
							|  |  |  | #[derive(Debug, Default)]
 | 
					
						
							|  |  |  | pub struct DataStore {
 | 
					
						
							| 
									
										
										
										
											2019-07-24 21:43:14 -07:00
										 |  |  |     data: IndexMap<AccessPath, Vec<u8>>,
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl DataStore {
 | 
					
						
							|  |  |  |     /// Creates a new `DataStore` with the provided initial data.
 | 
					
						
							| 
									
										
										
										
											2019-07-24 21:43:14 -07:00
										 |  |  |     pub fn new(data: IndexMap<AccessPath, Vec<u8>>) -> Self {
 | 
					
						
							| 
									
										
										
										
											2019-07-30 23:43:12 -07:00
										 |  |  |         Self { data }
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Applies a [`WriteSet`] to this data store.
 | 
					
						
							|  |  |  |     pub fn apply_write_set(&mut self, write_set: &WriteSet) {
 | 
					
						
							|  |  |  |         for (access_path, write_op) in write_set {
 | 
					
						
							|  |  |  |             match write_op {
 | 
					
						
							|  |  |  |                 WriteOp::Value(value) => {
 | 
					
						
							|  |  |  |                     self.set(access_path.clone(), value.clone());
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 WriteOp::Deletion => {
 | 
					
						
							|  |  |  |                     self.remove(access_path);
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Returns a `WriteSet` for each account in the `DataStore`
 | 
					
						
							| 
									
										
										
										
											2019-07-24 21:43:14 -07:00
										 |  |  |     pub fn into_write_sets(mut self) -> Result<IndexMap<AccountAddress, WriteSet>> {
 | 
					
						
							|  |  |  |         let mut write_set_muts: IndexMap<AccountAddress, WriteSetMut> = IndexMap::new();
 | 
					
						
							|  |  |  |         for (access_path, value) in self.data.drain(..) {
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |             match write_set_muts.get_mut(&access_path.address) {
 | 
					
						
							|  |  |  |                 Some(write_set_mut) => write_set_mut.push((access_path, WriteOp::Value(value))),
 | 
					
						
							|  |  |  |                 None => {
 | 
					
						
							|  |  |  |                     write_set_muts.insert(
 | 
					
						
							|  |  |  |                         access_path.address,
 | 
					
						
							|  |  |  |                         WriteSetMut::new(vec![(access_path, WriteOp::Value(value))]),
 | 
					
						
							|  |  |  |                     );
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         // Freeze each WriteSet
 | 
					
						
							| 
									
										
										
										
											2019-07-24 21:43:14 -07:00
										 |  |  |         let mut write_sets: IndexMap<AccountAddress, WriteSet> = IndexMap::new();
 | 
					
						
							|  |  |  |         for (address, write_set_mut) in write_set_muts.drain(..) {
 | 
					
						
							| 
									
										
										
										
											2019-07-24 12:30:43 -07:00
										 |  |  |             write_sets.insert(address, write_set_mut.freeze()?);
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2019-07-24 12:30:43 -07:00
										 |  |  |         Ok(write_sets)
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Read an account's resource
 | 
					
						
							| 
									
										
										
										
											2019-10-29 10:39:10 -07:00
										 |  |  |     pub fn read_account_resource(&self, addr: &AccountAddress) -> Option<AccountResource> {
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |         let access_path = create_access_path(&addr, account_config::account_struct_tag());
 | 
					
						
							| 
									
										
											  
											
												Maintenance : simplify a few patterns, remove unneeded dependencies (#8137)
* Simplify a few pattern matches
* Removing unneeded dependencies, upgrading internal version #s
 net-shaper: Removing log, semver, serde_derive
 bench-tps: Removing serde, serde_derive
 banking-bench: Removing solana
 ledger-tool: Removing bincode, serde, serde_derive
 librapay: Removing solana, language_e2e_tests
 log-analyzer: Removing log, semver, serde_derive
 exchange: Removing solana
 core: Removing crc, memmap, symlink, untrusted
 perf: Removing serde_derive
 genesis: Removing hex, serde_derive
 sdk-c: Removing sha2
 sys-tuner: Removing semver
 bench-exchange: Removing bincode, bs58, env_logger, serde, serde_derive, untrusted, ws
 btc_spv_bin: Removing serde_json
 btc_spv: Removing chrono
 bpf_loader: Removing serde
 ledger: Removing dlopen, dlopen_derive, serde_derive
 move_loader: Removing byteorder, libc, language_e2e_tests
 ownable: Removing serde, serde_derive
 client: Removing rand
 archiver-utils: Removing rand_chacha
 validator: Removing serde_json, tempfile
 param_passing_dep: Removing solana
 failure: Removing log
 vest: Removing log
 vote-signer: Removing bs58, serde
 local-cluster: Removing symlink
 keygen: Removing rpassword
 install: Removing bs58, log
 upload-perf: Removing log
 runtime: Removing serde_json
 stake: Removing rand
* Add modified Cargo.lock
* fixup! Simplify a few pattern matches
* fixup! Simplify a few pattern matches
											
										 
											2020-02-06 12:02:38 -05:00
										 |  |  |         self.data.get(&access_path).and_then(|blob| { SimpleDeserializer::deserialize(blob).ok() })
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Sets a (key, value) pair within this data store.
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// Returns the previous data if the key was occupied.
 | 
					
						
							|  |  |  |     pub fn set(&mut self, access_path: AccessPath, data_blob: Vec<u8>) -> Option<Vec<u8>> {
 | 
					
						
							|  |  |  |         self.data.insert(access_path, data_blob)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Deletes a key from this data store.
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// Returns the previous data if the key was occupied.
 | 
					
						
							|  |  |  |     pub fn remove(&mut self, access_path: &AccessPath) -> Option<Vec<u8>> {
 | 
					
						
							|  |  |  |         self.data.remove(access_path)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-18 16:47:01 -08:00
										 |  |  |     /// Adds a [`VerifiedModule`] to this data store.
 | 
					
						
							|  |  |  |     pub fn add_module(&mut self, module: &VerifiedModule) -> Result<()> {
 | 
					
						
							|  |  |  |         let access_path = AccessPath::from(&module.self_id());
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let mut bytes = vec![];
 | 
					
						
							|  |  |  |         module.serialize(&mut bytes)?;
 | 
					
						
							|  |  |  |         self.set(access_path, bytes);
 | 
					
						
							|  |  |  |         Ok(())
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Dumps the data store to stdout
 | 
					
						
							|  |  |  |     pub fn dump(&self) {
 | 
					
						
							| 
									
										
										
										
											2019-07-25 09:30:24 -07:00
										 |  |  |         trace!("Data store:");
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |         for (access_path, value) in &self.data {
 | 
					
						
							| 
									
										
										
										
											2019-07-24 21:43:14 -07:00
										 |  |  |             trace!("{:?}: {:?}", access_path, value.len());
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl StateView for DataStore {
 | 
					
						
							|  |  |  |     fn get(&self, access_path: &AccessPath) -> Result<Option<Vec<u8>>> {
 | 
					
						
							|  |  |  |         // Since the data is in-memory, it can't fail.
 | 
					
						
							|  |  |  |         match self.data.get(access_path) {
 | 
					
						
							|  |  |  |             None => Ok(None),
 | 
					
						
							|  |  |  |             Some(value) => Ok(Some(value.clone())),
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn multi_get(&self, _access_paths: &[AccessPath]) -> Result<Vec<Option<Vec<u8>>>> {
 | 
					
						
							|  |  |  |         unimplemented!();
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn is_genesis(&self) -> bool {
 | 
					
						
							|  |  |  |         false
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl RemoteCache for DataStore {
 | 
					
						
							| 
									
										
										
										
											2019-10-29 10:39:10 -07:00
										 |  |  |     fn get(&self, access_path: &AccessPath) -> VMResult<Option<Vec<u8>>> {
 | 
					
						
							| 
									
										
										
										
											2019-07-22 13:01:52 -06:00
										 |  |  |         Ok(StateView::get(self, access_path).expect("it should not error"))
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-24 21:43:14 -07:00
										 |  |  | #[cfg(test)]
 | 
					
						
							|  |  |  | mod tests {
 | 
					
						
							|  |  |  |     use super::*;
 | 
					
						
							|  |  |  |     use types::account_address::ADDRESS_LENGTH;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[test]
 | 
					
						
							|  |  |  |     fn test_write_set_order() {
 | 
					
						
							|  |  |  |         solana_logger::setup();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let mut data_store = DataStore::default();
 | 
					
						
							|  |  |  |         let address1 = AccountAddress::new([0; ADDRESS_LENGTH]);
 | 
					
						
							|  |  |  |         let address2 = AccountAddress::new([1; ADDRESS_LENGTH]);
 | 
					
						
							|  |  |  |         let address3 = AccountAddress::new([2; ADDRESS_LENGTH]);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let mut before1 = WriteSetMut::default();
 | 
					
						
							|  |  |  |         let mut before2 = WriteSetMut::default();
 | 
					
						
							|  |  |  |         let mut before3 = WriteSetMut::default();
 | 
					
						
							|  |  |  |         for i in 1..10 {
 | 
					
						
							|  |  |  |             before1.push((
 | 
					
						
							|  |  |  |                 AccessPath::new(address1, AccountAddress::random().to_vec()),
 | 
					
						
							|  |  |  |                 WriteOp::Value(vec![i]),
 | 
					
						
							|  |  |  |             ));
 | 
					
						
							|  |  |  |             before2.push((
 | 
					
						
							|  |  |  |                 AccessPath::new(address2, AccountAddress::random().to_vec()),
 | 
					
						
							|  |  |  |                 WriteOp::Value(vec![i]),
 | 
					
						
							|  |  |  |             ));
 | 
					
						
							|  |  |  |             before3.push((
 | 
					
						
							|  |  |  |                 AccessPath::new(address3, AccountAddress::random().to_vec()),
 | 
					
						
							|  |  |  |                 WriteOp::Value(vec![i]),
 | 
					
						
							|  |  |  |             ));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         let before1 = before1.freeze().unwrap();
 | 
					
						
							|  |  |  |         let before2 = before2.freeze().unwrap();
 | 
					
						
							|  |  |  |         let before3 = before3.freeze().unwrap();
 | 
					
						
							|  |  |  |         data_store.apply_write_set(&before1);
 | 
					
						
							|  |  |  |         data_store.apply_write_set(&before2);
 | 
					
						
							|  |  |  |         data_store.apply_write_set(&before3);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let write_sets = data_store.into_write_sets().unwrap();
 | 
					
						
							|  |  |  |         let after1 = write_sets.get(&address1).unwrap();
 | 
					
						
							|  |  |  |         let after2 = write_sets.get(&address2).unwrap();
 | 
					
						
							|  |  |  |         let after3 = write_sets.get(&address3).unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         assert_eq!(&before1, after1);
 | 
					
						
							|  |  |  |         assert_eq!(&before2, after2);
 | 
					
						
							|  |  |  |         assert_eq!(&before3, after3);
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2019-07-26 17:51:07 -07:00
										 |  |  | }
 |