2021-05-04 16:10:20 +08:00
|
|
|
use anyhow::Result;
|
2020-08-07 14:30:00 +01:00
|
|
|
use rand::prelude::*;
|
|
|
|
use std::collections::HashSet;
|
|
|
|
use std::fs::OpenOptions;
|
|
|
|
use std::path::Path;
|
2021-07-08 17:04:40 +08:00
|
|
|
use thinp::cache::ir::{self, MetadataVisitor};
|
2020-08-07 14:30:00 +01:00
|
|
|
use thinp::cache::xml;
|
|
|
|
|
|
|
|
//------------------------------------------
|
|
|
|
|
|
|
|
pub trait XmlGen {
|
2021-07-08 17:04:40 +08:00
|
|
|
fn generate_xml(&mut self, v: &mut dyn MetadataVisitor) -> Result<()>;
|
2020-08-07 14:30:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn write_xml(path: &Path, g: &mut dyn XmlGen) -> Result<()> {
|
|
|
|
let xml_out = OpenOptions::new()
|
|
|
|
.read(false)
|
|
|
|
.write(true)
|
|
|
|
.create(true)
|
|
|
|
.truncate(true)
|
|
|
|
.open(path)?;
|
|
|
|
let mut w = xml::XmlWriter::new(xml_out);
|
|
|
|
|
|
|
|
g.generate_xml(&mut w)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct CacheGen {
|
2021-02-25 18:14:01 +08:00
|
|
|
block_size: u32,
|
|
|
|
nr_cache_blocks: u32,
|
2020-08-07 14:30:00 +01:00
|
|
|
nr_origin_blocks: u64,
|
|
|
|
percent_resident: u8,
|
|
|
|
percent_dirty: u8,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CacheGen {
|
|
|
|
pub fn new(
|
2021-02-25 18:14:01 +08:00
|
|
|
block_size: u32,
|
|
|
|
nr_cache_blocks: u32,
|
2020-08-07 14:30:00 +01:00
|
|
|
nr_origin_blocks: u64,
|
|
|
|
percent_resident: u8,
|
|
|
|
percent_dirty: u8,
|
|
|
|
) -> Self {
|
|
|
|
CacheGen {
|
|
|
|
block_size,
|
|
|
|
nr_cache_blocks,
|
|
|
|
nr_origin_blocks,
|
|
|
|
percent_resident,
|
|
|
|
percent_dirty,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl XmlGen for CacheGen {
|
2021-07-08 17:04:40 +08:00
|
|
|
fn generate_xml(&mut self, v: &mut dyn MetadataVisitor) -> Result<()> {
|
|
|
|
v.superblock_b(&ir::Superblock {
|
2020-08-07 14:30:00 +01:00
|
|
|
uuid: "".to_string(),
|
|
|
|
block_size: self.block_size,
|
|
|
|
nr_cache_blocks: self.nr_cache_blocks,
|
|
|
|
policy: "smq".to_string(),
|
|
|
|
hint_width: 4,
|
|
|
|
})?;
|
|
|
|
|
2021-08-31 22:28:12 +08:00
|
|
|
let nr_resident = (self.nr_cache_blocks * self.percent_resident as u32) / 100u32;
|
|
|
|
let mut cblocks = (0..self.nr_cache_blocks).collect::<Vec<u32>>();
|
2020-08-07 14:30:00 +01:00
|
|
|
cblocks.shuffle(&mut rand::thread_rng());
|
2021-08-31 22:28:12 +08:00
|
|
|
cblocks.truncate(nr_resident as usize);
|
|
|
|
cblocks.sort();
|
2020-08-07 14:30:00 +01:00
|
|
|
|
|
|
|
v.mappings_b()?;
|
|
|
|
{
|
|
|
|
let mut used = HashSet::new();
|
2021-08-31 22:28:12 +08:00
|
|
|
let mut rng = rand::thread_rng();
|
|
|
|
for cblock in cblocks {
|
2020-08-07 14:30:00 +01:00
|
|
|
let mut oblock = 0u64;
|
|
|
|
while used.contains(&oblock) {
|
2021-08-31 22:28:12 +08:00
|
|
|
oblock = rng.gen_range(0..self.nr_origin_blocks);
|
2020-08-07 14:30:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
used.insert(oblock);
|
|
|
|
// FIXME: dirty should vary
|
2021-07-08 17:04:40 +08:00
|
|
|
v.mapping(&ir::Map {
|
2021-08-31 22:28:12 +08:00
|
|
|
cblock,
|
2020-08-07 14:30:00 +01:00
|
|
|
oblock,
|
|
|
|
dirty: false,
|
|
|
|
})?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
v.mappings_e()?;
|
|
|
|
|
|
|
|
v.superblock_e()?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------
|