| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  | //! this service receives instruction ExecuteTimings from replay_stage,
 | 
					
						
							|  |  |  | //! update cost_model which is shared with banking_stage to optimize
 | 
					
						
							|  |  |  | //! packing transactions into block; it also triggers persisting cost
 | 
					
						
							|  |  |  | //! table to blockstore.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 09:00:31 -08:00
										 |  |  | use {
 | 
					
						
							|  |  |  |     solana_ledger::blockstore::Blockstore,
 | 
					
						
							|  |  |  |     solana_measure::measure::Measure,
 | 
					
						
							| 
									
										
										
										
											2022-01-04 01:23:56 -07:00
										 |  |  |     solana_program_runtime::timings::ExecuteTimings,
 | 
					
						
							|  |  |  |     solana_runtime::{bank::Bank, cost_model::CostModel},
 | 
					
						
							| 
									
										
										
										
											2021-12-03 09:00:31 -08:00
										 |  |  |     solana_sdk::timing::timestamp,
 | 
					
						
							|  |  |  |     std::{
 | 
					
						
							|  |  |  |         sync::{
 | 
					
						
							|  |  |  |             atomic::{AtomicBool, Ordering},
 | 
					
						
							|  |  |  |             mpsc::Receiver,
 | 
					
						
							|  |  |  |             Arc, RwLock,
 | 
					
						
							|  |  |  |         },
 | 
					
						
							|  |  |  |         thread::{self, Builder, JoinHandle},
 | 
					
						
							|  |  |  |         time::Duration,
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |     },
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[derive(Default)]
 | 
					
						
							|  |  |  | pub struct CostUpdateServiceTiming {
 | 
					
						
							|  |  |  |     last_print: u64,
 | 
					
						
							|  |  |  |     update_cost_model_count: u64,
 | 
					
						
							|  |  |  |     update_cost_model_elapsed: u64,
 | 
					
						
							|  |  |  |     persist_cost_table_elapsed: u64,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl CostUpdateServiceTiming {
 | 
					
						
							|  |  |  |     fn update(
 | 
					
						
							|  |  |  |         &mut self,
 | 
					
						
							|  |  |  |         update_cost_model_count: u64,
 | 
					
						
							|  |  |  |         update_cost_model_elapsed: u64,
 | 
					
						
							|  |  |  |         persist_cost_table_elapsed: u64,
 | 
					
						
							|  |  |  |     ) {
 | 
					
						
							|  |  |  |         self.update_cost_model_count += update_cost_model_count;
 | 
					
						
							|  |  |  |         self.update_cost_model_elapsed += update_cost_model_elapsed;
 | 
					
						
							|  |  |  |         self.persist_cost_table_elapsed += persist_cost_table_elapsed;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let now = timestamp();
 | 
					
						
							|  |  |  |         let elapsed_ms = now - self.last_print;
 | 
					
						
							|  |  |  |         if elapsed_ms > 1000 {
 | 
					
						
							|  |  |  |             datapoint_info!(
 | 
					
						
							| 
									
										
										
										
											2021-09-15 09:49:49 -05:00
										 |  |  |                 "cost-update-service-stats",
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |                 ("total_elapsed_us", elapsed_ms * 1000, i64),
 | 
					
						
							|  |  |  |                 (
 | 
					
						
							|  |  |  |                     "update_cost_model_count",
 | 
					
						
							|  |  |  |                     self.update_cost_model_count as i64,
 | 
					
						
							|  |  |  |                     i64
 | 
					
						
							|  |  |  |                 ),
 | 
					
						
							|  |  |  |                 (
 | 
					
						
							|  |  |  |                     "update_cost_model_elapsed",
 | 
					
						
							|  |  |  |                     self.update_cost_model_elapsed as i64,
 | 
					
						
							|  |  |  |                     i64
 | 
					
						
							|  |  |  |                 ),
 | 
					
						
							|  |  |  |                 (
 | 
					
						
							|  |  |  |                     "persist_cost_table_elapsed",
 | 
					
						
							|  |  |  |                     self.persist_cost_table_elapsed as i64,
 | 
					
						
							|  |  |  |                     i64
 | 
					
						
							|  |  |  |                 ),
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             *self = CostUpdateServiceTiming::default();
 | 
					
						
							|  |  |  |             self.last_print = now;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-24 22:19:23 -05:00
										 |  |  | pub enum CostUpdate {
 | 
					
						
							|  |  |  |     FrozenBank { bank: Arc<Bank> },
 | 
					
						
							|  |  |  |     ExecuteTiming { execute_timings: ExecuteTimings },
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub type CostUpdateReceiver = Receiver<CostUpdate>;
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | pub struct CostUpdateService {
 | 
					
						
							|  |  |  |     thread_hdl: JoinHandle<()>,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl CostUpdateService {
 | 
					
						
							|  |  |  |     #[allow(clippy::new_ret_no_self)]
 | 
					
						
							|  |  |  |     pub fn new(
 | 
					
						
							|  |  |  |         exit: Arc<AtomicBool>,
 | 
					
						
							|  |  |  |         blockstore: Arc<Blockstore>,
 | 
					
						
							|  |  |  |         cost_model: Arc<RwLock<CostModel>>,
 | 
					
						
							|  |  |  |         cost_update_receiver: CostUpdateReceiver,
 | 
					
						
							|  |  |  |     ) -> Self {
 | 
					
						
							|  |  |  |         let thread_hdl = Builder::new()
 | 
					
						
							|  |  |  |             .name("solana-cost-update-service".to_string())
 | 
					
						
							|  |  |  |             .spawn(move || {
 | 
					
						
							|  |  |  |                 Self::service_loop(exit, blockstore, cost_model, cost_update_receiver);
 | 
					
						
							|  |  |  |             })
 | 
					
						
							|  |  |  |             .unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Self { thread_hdl }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn join(self) -> thread::Result<()> {
 | 
					
						
							|  |  |  |         self.thread_hdl.join()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn service_loop(
 | 
					
						
							|  |  |  |         exit: Arc<AtomicBool>,
 | 
					
						
							|  |  |  |         blockstore: Arc<Blockstore>,
 | 
					
						
							|  |  |  |         cost_model: Arc<RwLock<CostModel>>,
 | 
					
						
							|  |  |  |         cost_update_receiver: CostUpdateReceiver,
 | 
					
						
							|  |  |  |     ) {
 | 
					
						
							|  |  |  |         let mut cost_update_service_timing = CostUpdateServiceTiming::default();
 | 
					
						
							| 
									
										
										
										
											2021-09-15 09:49:49 -05:00
										 |  |  |         let mut dirty: bool;
 | 
					
						
							|  |  |  |         let mut update_count: u64;
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |         let wait_timer = Duration::from_millis(100);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop {
 | 
					
						
							|  |  |  |             if exit.load(Ordering::Relaxed) {
 | 
					
						
							|  |  |  |                 break;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 09:49:49 -05:00
										 |  |  |             dirty = false;
 | 
					
						
							|  |  |  |             update_count = 0_u64;
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |             let mut update_cost_model_time = Measure::start("update_cost_model_time");
 | 
					
						
							|  |  |  |             for cost_update in cost_update_receiver.try_iter() {
 | 
					
						
							| 
									
										
										
										
											2021-10-24 22:19:23 -05:00
										 |  |  |                 match cost_update {
 | 
					
						
							|  |  |  |                     CostUpdate::FrozenBank { bank } => {
 | 
					
						
							|  |  |  |                         bank.read_cost_tracker().unwrap().report_stats(bank.slot());
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |                     CostUpdate::ExecuteTiming {
 | 
					
						
							|  |  |  |                         mut execute_timings,
 | 
					
						
							|  |  |  |                     } => {
 | 
					
						
							|  |  |  |                         dirty |= Self::update_cost_model(&cost_model, &mut execute_timings);
 | 
					
						
							| 
									
										
										
										
											2021-10-24 22:19:23 -05:00
										 |  |  |                         update_count += 1;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |             }
 | 
					
						
							|  |  |  |             update_cost_model_time.stop();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let mut persist_cost_table_time = Measure::start("persist_cost_table_time");
 | 
					
						
							|  |  |  |             if dirty {
 | 
					
						
							|  |  |  |                 Self::persist_cost_table(&blockstore, &cost_model);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             persist_cost_table_time.stop();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             cost_update_service_timing.update(
 | 
					
						
							|  |  |  |                 update_count,
 | 
					
						
							|  |  |  |                 update_cost_model_time.as_us(),
 | 
					
						
							|  |  |  |                 persist_cost_table_time.as_us(),
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             thread::sleep(wait_timer);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |     fn update_cost_model(
 | 
					
						
							|  |  |  |         cost_model: &RwLock<CostModel>,
 | 
					
						
							|  |  |  |         execute_timings: &mut ExecuteTimings,
 | 
					
						
							|  |  |  |     ) -> bool {
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |         let mut dirty = false;
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |         {
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |             for (program_id, program_timings) in &mut execute_timings.details.per_program_timings {
 | 
					
						
							|  |  |  |                 let current_estimated_program_cost =
 | 
					
						
							|  |  |  |                     cost_model.read().unwrap().find_instruction_cost(program_id);
 | 
					
						
							|  |  |  |                 program_timings.coalesce_error_timings(current_estimated_program_cost);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if program_timings.count < 1 {
 | 
					
						
							| 
									
										
										
										
											2021-07-29 17:19:36 -05:00
										 |  |  |                     continue;
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 let units = program_timings.accumulated_units / program_timings.count as u64;
 | 
					
						
							|  |  |  |                 match cost_model
 | 
					
						
							|  |  |  |                     .write()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .upsert_instruction_cost(program_id, units)
 | 
					
						
							|  |  |  |                 {
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |                     Ok(c) => {
 | 
					
						
							|  |  |  |                         debug!(
 | 
					
						
							|  |  |  |                             "after replayed into bank, instruction {:?} has averaged cost {}",
 | 
					
						
							|  |  |  |                             program_id, c
 | 
					
						
							|  |  |  |                         );
 | 
					
						
							|  |  |  |                         dirty = true;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                     Err(err) => {
 | 
					
						
							|  |  |  |                         debug!(
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |                         "after replayed into bank, instruction {:?} failed to update cost, err: {}",
 | 
					
						
							|  |  |  |                         program_id, err
 | 
					
						
							|  |  |  |                     );
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |                     }
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         debug!(
 | 
					
						
							|  |  |  |            "after replayed into bank, updated cost model instruction cost table, current values: {:?}",
 | 
					
						
							|  |  |  |            cost_model.read().unwrap().get_instruction_cost_table()
 | 
					
						
							|  |  |  |         );
 | 
					
						
							|  |  |  |         dirty
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn persist_cost_table(blockstore: &Blockstore, cost_model: &RwLock<CostModel>) {
 | 
					
						
							|  |  |  |         let cost_model_read = cost_model.read().unwrap();
 | 
					
						
							|  |  |  |         let cost_table = cost_model_read.get_instruction_cost_table();
 | 
					
						
							|  |  |  |         let db_records = blockstore.read_program_costs().expect("read programs");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // delete records from blockstore if they are no longer in cost_table
 | 
					
						
							|  |  |  |         db_records.iter().for_each(|(pubkey, _)| {
 | 
					
						
							|  |  |  |             if cost_table.get(pubkey).is_none() {
 | 
					
						
							|  |  |  |                 blockstore
 | 
					
						
							|  |  |  |                     .delete_program_cost(pubkey)
 | 
					
						
							|  |  |  |                     .expect("delete old program");
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         });
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (key, cost) in cost_table.iter() {
 | 
					
						
							|  |  |  |             blockstore
 | 
					
						
							|  |  |  |                 .write_program_cost(key, cost)
 | 
					
						
							|  |  |  |                 .expect("persist program costs to blockstore");
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[cfg(test)]
 | 
					
						
							|  |  |  | mod tests {
 | 
					
						
							| 
									
										
										
										
											2021-12-03 09:00:31 -08:00
										 |  |  |     use {super::*, solana_program_runtime::timings::ProgramTiming, solana_sdk::pubkey::Pubkey};
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     #[test]
 | 
					
						
							|  |  |  |     fn test_update_cost_model_with_empty_execute_timings() {
 | 
					
						
							|  |  |  |         let cost_model = Arc::new(RwLock::new(CostModel::default()));
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |         let mut empty_execute_timings = ExecuteTimings::default();
 | 
					
						
							|  |  |  |         CostUpdateService::update_cost_model(&cost_model, &mut empty_execute_timings);
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         assert_eq!(
 | 
					
						
							|  |  |  |             0,
 | 
					
						
							|  |  |  |             cost_model
 | 
					
						
							|  |  |  |                 .read()
 | 
					
						
							|  |  |  |                 .unwrap()
 | 
					
						
							|  |  |  |                 .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                 .len()
 | 
					
						
							|  |  |  |         );
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[test]
 | 
					
						
							|  |  |  |     fn test_update_cost_model_with_execute_timings() {
 | 
					
						
							|  |  |  |         let cost_model = Arc::new(RwLock::new(CostModel::default()));
 | 
					
						
							|  |  |  |         let mut execute_timings = ExecuteTimings::default();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let program_key_1 = Pubkey::new_unique();
 | 
					
						
							|  |  |  |         let mut expected_cost: u64;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // add new program
 | 
					
						
							|  |  |  |         {
 | 
					
						
							|  |  |  |             let accumulated_us: u64 = 1000;
 | 
					
						
							| 
									
										
										
										
											2021-07-16 12:40:12 -07:00
										 |  |  |             let accumulated_units: u64 = 100;
 | 
					
						
							| 
									
										
										
										
											2022-01-03 16:33:54 -05:00
										 |  |  |             let total_errored_units = 0;
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |             let count: u32 = 10;
 | 
					
						
							| 
									
										
										
										
											2021-07-29 17:19:36 -05:00
										 |  |  |             expected_cost = accumulated_units / count as u64;
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |             execute_timings.details.per_program_timings.insert(
 | 
					
						
							|  |  |  |                 program_key_1,
 | 
					
						
							|  |  |  |                 ProgramTiming {
 | 
					
						
							|  |  |  |                     accumulated_us,
 | 
					
						
							| 
									
										
										
										
											2021-07-16 12:40:12 -07:00
										 |  |  |                     accumulated_units,
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |                     count,
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |                     errored_txs_compute_consumed: vec![],
 | 
					
						
							| 
									
										
										
										
											2022-01-03 16:33:54 -05:00
										 |  |  |                     total_errored_units,
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |                 },
 | 
					
						
							|  |  |  |             );
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |             CostUpdateService::update_cost_model(&cost_model, &mut execute_timings);
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |             assert_eq!(
 | 
					
						
							|  |  |  |                 1,
 | 
					
						
							|  |  |  |                 cost_model
 | 
					
						
							|  |  |  |                     .read()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                     .len()
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |             assert_eq!(
 | 
					
						
							|  |  |  |                 Some(&expected_cost),
 | 
					
						
							|  |  |  |                 cost_model
 | 
					
						
							|  |  |  |                     .read()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                     .get(&program_key_1)
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // update program
 | 
					
						
							|  |  |  |         {
 | 
					
						
							|  |  |  |             let accumulated_us: u64 = 2000;
 | 
					
						
							| 
									
										
										
										
											2021-07-16 12:40:12 -07:00
										 |  |  |             let accumulated_units: u64 = 200;
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |             let count: u32 = 10;
 | 
					
						
							|  |  |  |             // to expect new cost is Average(new_value, existing_value)
 | 
					
						
							| 
									
										
										
										
											2021-07-29 17:19:36 -05:00
										 |  |  |             expected_cost = ((accumulated_units / count as u64) + expected_cost) / 2;
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |             execute_timings.details.per_program_timings.insert(
 | 
					
						
							|  |  |  |                 program_key_1,
 | 
					
						
							|  |  |  |                 ProgramTiming {
 | 
					
						
							|  |  |  |                     accumulated_us,
 | 
					
						
							| 
									
										
										
										
											2021-07-16 12:40:12 -07:00
										 |  |  |                     accumulated_units,
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |                     count,
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |                     errored_txs_compute_consumed: vec![],
 | 
					
						
							| 
									
										
										
										
											2022-01-03 16:33:54 -05:00
										 |  |  |                     total_errored_units: 0,
 | 
					
						
							| 
									
										
										
										
											2021-07-09 12:55:31 -07:00
										 |  |  |                 },
 | 
					
						
							|  |  |  |             );
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |             CostUpdateService::update_cost_model(&cost_model, &mut execute_timings);
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  |             assert_eq!(
 | 
					
						
							|  |  |  |                 1,
 | 
					
						
							|  |  |  |                 cost_model
 | 
					
						
							|  |  |  |                     .read()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                     .len()
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |             assert_eq!(
 | 
					
						
							|  |  |  |                 Some(&expected_cost),
 | 
					
						
							|  |  |  |                 cost_model
 | 
					
						
							|  |  |  |                     .read()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                     .get(&program_key_1)
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     #[test]
 | 
					
						
							|  |  |  |     fn test_update_cost_model_with_error_execute_timings() {
 | 
					
						
							|  |  |  |         let cost_model = Arc::new(RwLock::new(CostModel::default()));
 | 
					
						
							|  |  |  |         let mut execute_timings = ExecuteTimings::default();
 | 
					
						
							|  |  |  |         let program_key_1 = Pubkey::new_unique();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Test updating cost model with a `ProgramTiming` with no compute units accumulated, i.e.
 | 
					
						
							|  |  |  |         // `accumulated_units` == 0
 | 
					
						
							|  |  |  |         {
 | 
					
						
							|  |  |  |             execute_timings.details.per_program_timings.insert(
 | 
					
						
							|  |  |  |                 program_key_1,
 | 
					
						
							|  |  |  |                 ProgramTiming {
 | 
					
						
							|  |  |  |                     accumulated_us: 1000,
 | 
					
						
							|  |  |  |                     accumulated_units: 0,
 | 
					
						
							|  |  |  |                     count: 0,
 | 
					
						
							|  |  |  |                     errored_txs_compute_consumed: vec![],
 | 
					
						
							| 
									
										
										
										
											2022-01-03 16:33:54 -05:00
										 |  |  |                     total_errored_units: 0,
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |                 },
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |             CostUpdateService::update_cost_model(&cost_model, &mut execute_timings);
 | 
					
						
							|  |  |  |             // If both the `errored_txs_compute_consumed` is empty and `count == 0`, then
 | 
					
						
							|  |  |  |             // nothing should be inserted into the cost model
 | 
					
						
							|  |  |  |             assert!(cost_model
 | 
					
						
							|  |  |  |                 .read()
 | 
					
						
							|  |  |  |                 .unwrap()
 | 
					
						
							|  |  |  |                 .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                 .is_empty());
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Test updating cost model with only erroring compute costs where the `cost_per_error` is
 | 
					
						
							|  |  |  |         // greater than the current instruction cost for the program. Should update with the
 | 
					
						
							|  |  |  |         // new erroring compute costs
 | 
					
						
							|  |  |  |         let cost_per_error = 1000;
 | 
					
						
							|  |  |  |         {
 | 
					
						
							| 
									
										
										
										
											2022-01-03 16:33:54 -05:00
										 |  |  |             let errored_txs_compute_consumed = vec![cost_per_error; 3];
 | 
					
						
							|  |  |  |             let total_errored_units = errored_txs_compute_consumed.iter().sum();
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |             execute_timings.details.per_program_timings.insert(
 | 
					
						
							|  |  |  |                 program_key_1,
 | 
					
						
							|  |  |  |                 ProgramTiming {
 | 
					
						
							|  |  |  |                     accumulated_us: 1000,
 | 
					
						
							|  |  |  |                     accumulated_units: 0,
 | 
					
						
							|  |  |  |                     count: 0,
 | 
					
						
							| 
									
										
										
										
											2022-01-03 16:33:54 -05:00
										 |  |  |                     errored_txs_compute_consumed,
 | 
					
						
							|  |  |  |                     total_errored_units,
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |                 },
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |             CostUpdateService::update_cost_model(&cost_model, &mut execute_timings);
 | 
					
						
							|  |  |  |             assert_eq!(
 | 
					
						
							|  |  |  |                 1,
 | 
					
						
							|  |  |  |                 cost_model
 | 
					
						
							|  |  |  |                     .read()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                     .len()
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |             assert_eq!(
 | 
					
						
							|  |  |  |                 Some(&cost_per_error),
 | 
					
						
							|  |  |  |                 cost_model
 | 
					
						
							|  |  |  |                     .read()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                     .get(&program_key_1)
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Test updating cost model with only erroring compute costs where the error cost is
 | 
					
						
							|  |  |  |         // `smaller_cost_per_error`, less than the current instruction cost for the program.
 | 
					
						
							|  |  |  |         // The cost should not decrease for these new lesser errors
 | 
					
						
							|  |  |  |         let smaller_cost_per_error = cost_per_error - 10;
 | 
					
						
							|  |  |  |         {
 | 
					
						
							| 
									
										
										
										
											2022-01-03 16:33:54 -05:00
										 |  |  |             let errored_txs_compute_consumed = vec![smaller_cost_per_error; 3];
 | 
					
						
							|  |  |  |             let total_errored_units = errored_txs_compute_consumed.iter().sum();
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |             execute_timings.details.per_program_timings.insert(
 | 
					
						
							|  |  |  |                 program_key_1,
 | 
					
						
							|  |  |  |                 ProgramTiming {
 | 
					
						
							|  |  |  |                     accumulated_us: 1000,
 | 
					
						
							|  |  |  |                     accumulated_units: 0,
 | 
					
						
							|  |  |  |                     count: 0,
 | 
					
						
							| 
									
										
										
										
											2022-01-03 16:33:54 -05:00
										 |  |  |                     errored_txs_compute_consumed,
 | 
					
						
							|  |  |  |                     total_errored_units,
 | 
					
						
							| 
									
										
										
										
											2021-12-30 21:21:42 -05:00
										 |  |  |                 },
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |             CostUpdateService::update_cost_model(&cost_model, &mut execute_timings);
 | 
					
						
							|  |  |  |             assert_eq!(
 | 
					
						
							|  |  |  |                 1,
 | 
					
						
							|  |  |  |                 cost_model
 | 
					
						
							|  |  |  |                     .read()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                     .len()
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |             assert_eq!(
 | 
					
						
							|  |  |  |                 Some(&cost_per_error),
 | 
					
						
							|  |  |  |                 cost_model
 | 
					
						
							|  |  |  |                     .read()
 | 
					
						
							|  |  |  |                     .unwrap()
 | 
					
						
							|  |  |  |                     .get_instruction_cost_table()
 | 
					
						
							|  |  |  |                     .get(&program_key_1)
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2021-07-01 11:32:41 -05:00
										 |  |  | }
 |