[thin_shrink] add some more test cases
This commit is contained in:
parent
3618da3a12
commit
4d7f9c7ddc
@ -20,7 +20,7 @@ where
|
|||||||
{
|
{
|
||||||
let mut buf = vec![0; len];
|
let mut buf = vec![0; len];
|
||||||
file.seek(SeekFrom::Start(src_byte))?;
|
file.seek(SeekFrom::Start(src_byte))?;
|
||||||
file.read_exact(&mut buf[0..])?;
|
file.read_exact(&mut buf)?;
|
||||||
file.seek(SeekFrom::Start(dest_byte))?;
|
file.seek(SeekFrom::Start(dest_byte))?;
|
||||||
file.write_all(&buf)?;
|
file.write_all(&buf)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -55,6 +55,7 @@ pub fn copy(path: &Path, regions: &[Region]) -> Result<()> {
|
|||||||
eprintln!("copying {:?}", r);
|
eprintln!("copying {:?}", r);
|
||||||
copy_region(&mut input, r)?;
|
copy_region(&mut input, r)?;
|
||||||
}
|
}
|
||||||
|
input.flush()?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use anyhow::Result;
|
use anyhow::{anyhow, Result};
|
||||||
use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt};
|
use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt};
|
||||||
use rand::Rng;
|
use rand::Rng;
|
||||||
use std::fs::OpenOptions;
|
use std::fs::OpenOptions;
|
||||||
@ -11,6 +11,7 @@ use thinp::thin::xml::{self, Visit};
|
|||||||
|
|
||||||
//------------------------------------
|
//------------------------------------
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
struct ThinBlock {
|
struct ThinBlock {
|
||||||
thin_id: u32,
|
thin_id: u32,
|
||||||
thin_block: u64,
|
thin_block: u64,
|
||||||
@ -33,35 +34,36 @@ impl ThinBlock {
|
|||||||
let mut rr = ThinReadRef {
|
let mut rr = ThinReadRef {
|
||||||
data: vec![0; self.block_size],
|
data: vec![0; self.block_size],
|
||||||
};
|
};
|
||||||
r.seek(SeekFrom::Start(self.data_block * (self.block_size as u64)))?;
|
let byte = self.data_block * (self.block_size as u64) * 512;
|
||||||
r.read_exact(&mut rr.data[0..])?;
|
r.seek(SeekFrom::Start(byte))?;
|
||||||
|
r.read_exact(&mut rr.data)?;
|
||||||
Ok(rr)
|
Ok(rr)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn zero_ref<'a, W: Write + Seek>(&self, w: &'a mut W) -> ThinWriteRef<'a, W> {
|
fn zero_ref<'a, W: Write + Seek>(&self, w: &'a mut W) -> ThinWriteRef<'a, W> {
|
||||||
ThinWriteRef {
|
ThinWriteRef {
|
||||||
file: w,
|
file: w,
|
||||||
block_byte: self.data_block * (self.block_size as u64),
|
block_byte: self.data_block * (self.block_size as u64) * 512,
|
||||||
data: vec![0; self.block_size],
|
data: vec![0; self.block_size],
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_ref<'a, W>(&self, w: &'a mut W) -> Result<ThinWriteRef<'a, W>>
|
//fn write_ref<'a, W>(&self, w: &'a mut W) -> Result<ThinWriteRef<'a, W>>
|
||||||
where
|
//where
|
||||||
W: Read + Write + Seek,
|
//W: Read + Write + Seek,
|
||||||
{
|
//{
|
||||||
let mut data = vec![0; self.block_size];
|
//let mut data = vec![0; self.block_size];
|
||||||
w.seek(SeekFrom::Start(self.data_block * (self.block_size as u64)))?;
|
//w.seek(SeekFrom::Start(self.data_block * (self.block_size as u64)))?;
|
||||||
w.read_exact(&mut data[0..])?;
|
//w.read_exact(&mut data[0..])?;
|
||||||
|
//
|
||||||
let wr = ThinWriteRef {
|
//let wr = ThinWriteRef {
|
||||||
file: w,
|
//file: w,
|
||||||
block_byte: self.data_block * (self.block_size as u64),
|
//block_byte: self.data_block * (self.block_size as u64),
|
||||||
data: vec![0; self.block_size],
|
//data: vec![0; self.block_size],
|
||||||
};
|
//};
|
||||||
|
//
|
||||||
Ok(wr)
|
//Ok(wr)
|
||||||
}
|
//}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, W: Write + Seek> Drop for ThinWriteRef<'a, W> {
|
impl<'a, W: Write + Seek> Drop for ThinWriteRef<'a, W> {
|
||||||
@ -70,7 +72,7 @@ impl<'a, W: Write + Seek> Drop for ThinWriteRef<'a, W> {
|
|||||||
// errors will have to make their way back to the user
|
// errors will have to make their way back to the user
|
||||||
// another way (eg, via a flush() method).
|
// another way (eg, via a flush() method).
|
||||||
self.file.seek(SeekFrom::Start(self.block_byte)).unwrap();
|
self.file.seek(SeekFrom::Start(self.block_byte)).unwrap();
|
||||||
self.file.write_all(&self.data[0..]).unwrap();
|
self.file.write_all(&self.data).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -161,34 +163,41 @@ impl Generator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn step(&mut self) {
|
fn step(&mut self) {
|
||||||
self.x = (self.a * self.x) + self.c
|
self.x = self.a.wrapping_mul(self.x).wrapping_add(self.c)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn fill_buffer(&mut self, seed: u64, bytes: &mut [u8]) {
|
fn fill_buffer(&mut self, seed: u64, bytes: &mut [u8]) -> Result<()> {
|
||||||
self.x = seed;
|
self.x = seed;
|
||||||
|
|
||||||
assert!(bytes.len() % 8 == 64);
|
assert!(bytes.len() % 8 == 0);
|
||||||
let nr_words = bytes.len() / 8;
|
let nr_words = bytes.len() / 8;
|
||||||
let mut out = Cursor::new(bytes);
|
let mut out = Cursor::new(bytes);
|
||||||
|
|
||||||
for _ in 0..nr_words {
|
for _ in 0..nr_words {
|
||||||
out.write_u64::<LittleEndian>(self.x).unwrap();
|
out.write_u64::<LittleEndian>(self.x)?;
|
||||||
self.step();
|
self.step();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn verify_buffer(&mut self, seed: u64, bytes: &[u8]) {
|
fn verify_buffer(&mut self, seed: u64, bytes: &[u8]) -> Result<bool> {
|
||||||
self.x = seed;
|
self.x = seed;
|
||||||
|
|
||||||
assert!(bytes.len() % 8 == 64);
|
assert!(bytes.len() % 8 == 0);
|
||||||
let nr_words = bytes.len() / 8;
|
let nr_words = bytes.len() / 8;
|
||||||
let mut input = Cursor::new(bytes);
|
let mut input = Cursor::new(bytes);
|
||||||
|
|
||||||
for _ in 0..nr_words {
|
for _ in 0..nr_words {
|
||||||
let w = input.read_u64::<LittleEndian>().unwrap();
|
let w = input.read_u64::<LittleEndian>()?;
|
||||||
assert_eq!(w, self.x);
|
if w != self.x {
|
||||||
|
eprintln!("{} != {}", w, self.x);
|
||||||
|
return Ok(false);
|
||||||
|
}
|
||||||
self.step();
|
self.step();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Ok(true)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -212,7 +221,7 @@ impl<'a, W: Write + Seek> ThinVisitor for Stamper<'a, W> {
|
|||||||
gen.fill_buffer(
|
gen.fill_buffer(
|
||||||
self.seed ^ (b.thin_id as u64) ^ b.thin_block,
|
self.seed ^ (b.thin_id as u64) ^ b.thin_block,
|
||||||
&mut wr.data[0..],
|
&mut wr.data[0..],
|
||||||
);
|
)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -234,7 +243,9 @@ impl<'a, R: Read + Seek> ThinVisitor for Verifier<'a, R> {
|
|||||||
fn thin_block(&mut self, b: &ThinBlock) -> Result<()> {
|
fn thin_block(&mut self, b: &ThinBlock) -> Result<()> {
|
||||||
let rr = b.read_ref(self.data_file)?;
|
let rr = b.read_ref(self.data_file)?;
|
||||||
let mut gen = Generator::new();
|
let mut gen = Generator::new();
|
||||||
gen.verify_buffer(self.seed ^ (b.thin_id as u64) ^ b.thin_block, &rr.data[0..]);
|
if !gen.verify_buffer(self.seed ^ (b.thin_id as u64) ^ b.thin_block, &rr.data[0..])? {
|
||||||
|
return Err(anyhow!("data verify failed for {:?}", b));
|
||||||
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -248,10 +259,7 @@ fn mk_path(dir: &Path, file: &str) -> PathBuf {
|
|||||||
p
|
p
|
||||||
}
|
}
|
||||||
|
|
||||||
fn generate_xml<G>(path: &Path, g: &mut G) -> Result<()>
|
fn generate_xml(path: &Path, g: &mut dyn Scenario) -> Result<()> {
|
||||||
where
|
|
||||||
G: XmlGenerator,
|
|
||||||
{
|
|
||||||
let xml_out = OpenOptions::new()
|
let xml_out = OpenOptions::new()
|
||||||
.read(false)
|
.read(false)
|
||||||
.write(true)
|
.write(true)
|
||||||
@ -260,7 +268,7 @@ where
|
|||||||
.open(path)?;
|
.open(path)?;
|
||||||
let mut w = xml::XmlWriter::new(xml_out);
|
let mut w = xml::XmlWriter::new(xml_out);
|
||||||
|
|
||||||
g.generate(&mut w)
|
g.generate_xml(&mut w)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_data_file(data_path: &Path, xml_path: &Path) -> Result<()> {
|
fn create_data_file(data_path: &Path, xml_path: &Path) -> Result<()> {
|
||||||
@ -296,50 +304,135 @@ fn verify(xml_path: &Path, data_path: &Path, seed: u64) -> Result<()> {
|
|||||||
thin_visit(xml, &mut verifier)
|
thin_visit(xml, &mut verifier)
|
||||||
}
|
}
|
||||||
|
|
||||||
//------------------------------------
|
trait Scenario {
|
||||||
|
fn generate_xml(&mut self, v: &mut dyn xml::MetadataVisitor) -> Result<()>;
|
||||||
trait XmlGenerator {
|
fn get_new_nr_blocks(&self) -> u64;
|
||||||
fn generate(&mut self, v: &mut dyn xml::MetadataVisitor) -> Result<()>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct EmptyPoolG {}
|
fn test_shrink(scenario: &mut dyn Scenario) -> Result<()> {
|
||||||
|
|
||||||
impl XmlGenerator for EmptyPoolG {
|
|
||||||
fn generate(&mut self, v: &mut dyn xml::MetadataVisitor) -> Result<()> {
|
|
||||||
v.superblock_b(&xml::Superblock {
|
|
||||||
uuid: "".to_string(),
|
|
||||||
time: 0,
|
|
||||||
transaction: 0,
|
|
||||||
flags: None,
|
|
||||||
version: None,
|
|
||||||
data_block_size: 64,
|
|
||||||
nr_data_blocks: 1024,
|
|
||||||
metadata_snap: None,
|
|
||||||
})?;
|
|
||||||
v.superblock_e()?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn shrink_empty_pool() -> Result<()> {
|
|
||||||
let dir = tempdir()?;
|
let dir = tempdir()?;
|
||||||
let xml_before = mk_path(dir.path(), "before.xml");
|
let xml_before = mk_path(dir.path(), "before.xml");
|
||||||
let xml_after = mk_path(dir.path(), "after.xml");
|
let xml_after = mk_path(dir.path(), "after.xml");
|
||||||
let data_path = mk_path(dir.path(), "bin");
|
let data_path = mk_path(dir.path(), "metadata.bin");
|
||||||
|
|
||||||
let mut gen = EmptyPoolG {};
|
generate_xml(&xml_before, scenario)?;
|
||||||
generate_xml(&xml_before, &mut gen)?;
|
|
||||||
create_data_file(&data_path, &xml_before)?;
|
create_data_file(&data_path, &xml_before)?;
|
||||||
|
|
||||||
let mut rng = rand::thread_rng();
|
let mut rng = rand::thread_rng();
|
||||||
let seed = rng.gen::<u64>();
|
let seed = rng.gen::<u64>();
|
||||||
|
|
||||||
stamp(&xml_before, &data_path, seed)?;
|
stamp(&xml_before, &data_path, seed)?;
|
||||||
|
|
||||||
let new_nr_blocks = 10;
|
let new_nr_blocks = scenario.get_new_nr_blocks();
|
||||||
thinp::shrink::toplevel::shrink(&xml_before, &xml_after, &data_path, new_nr_blocks, true)?;
|
thinp::shrink::toplevel::shrink(&xml_before, &xml_after, &data_path, new_nr_blocks, true)?;
|
||||||
|
|
||||||
verify(&xml_after, &data_path, seed)?;
|
verify(&xml_after, &data_path, seed)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//------------------------------------
|
||||||
|
|
||||||
|
fn common_sb(nr_blocks: u64) -> xml::Superblock {
|
||||||
|
xml::Superblock {
|
||||||
|
uuid: "".to_string(),
|
||||||
|
time: 0,
|
||||||
|
transaction: 0,
|
||||||
|
flags: None,
|
||||||
|
version: None,
|
||||||
|
data_block_size: 32,
|
||||||
|
nr_data_blocks: nr_blocks,
|
||||||
|
metadata_snap: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct EmptyPoolS {}
|
||||||
|
|
||||||
|
impl Scenario for EmptyPoolS {
|
||||||
|
fn generate_xml(&mut self, v: &mut dyn xml::MetadataVisitor) -> Result<()> {
|
||||||
|
v.superblock_b(&common_sb(1024))?;
|
||||||
|
v.superblock_e()?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_new_nr_blocks(&self) -> u64 {
|
||||||
|
512
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn shrink_empty_pool() -> Result<()> {
|
||||||
|
let mut s = EmptyPoolS {};
|
||||||
|
test_shrink(&mut s)
|
||||||
|
}
|
||||||
|
|
||||||
|
//------------------------------------
|
||||||
|
|
||||||
|
struct SingleThinS {
|
||||||
|
offset: u64,
|
||||||
|
len: u64,
|
||||||
|
old_nr_data_blocks: u64,
|
||||||
|
new_nr_data_blocks: u64,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SingleThinS {
|
||||||
|
fn new(offset: u64, len: u64, old_nr_data_blocks: u64, new_nr_data_blocks: u64) -> Self {
|
||||||
|
SingleThinS {
|
||||||
|
offset,
|
||||||
|
len,
|
||||||
|
old_nr_data_blocks,
|
||||||
|
new_nr_data_blocks,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Scenario for SingleThinS {
|
||||||
|
fn generate_xml(&mut self, v: &mut dyn xml::MetadataVisitor) -> Result<()> {
|
||||||
|
v.superblock_b(&common_sb(self.old_nr_data_blocks))?;
|
||||||
|
v.device_b(&xml::Device {
|
||||||
|
dev_id: 0,
|
||||||
|
mapped_blocks: self.len,
|
||||||
|
transaction: 0,
|
||||||
|
creation_time: 0,
|
||||||
|
snap_time: 0,
|
||||||
|
})?;
|
||||||
|
v.map(&xml::Map {
|
||||||
|
thin_begin: 0,
|
||||||
|
data_begin: self.offset,
|
||||||
|
time: 0,
|
||||||
|
len: self.len,
|
||||||
|
})?;
|
||||||
|
v.device_e()?;
|
||||||
|
v.superblock_e()?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_new_nr_blocks(&self) -> u64 {
|
||||||
|
self.new_nr_data_blocks
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn shrink_single_no_move_1() -> Result<()> {
|
||||||
|
let mut s = SingleThinS::new(0, 1024, 2048, 1280);
|
||||||
|
test_shrink(&mut s)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn shrink_single_no_move_2() -> Result<()> {
|
||||||
|
let mut s = SingleThinS::new(100, 1024, 2048, 1280);
|
||||||
|
test_shrink(&mut s)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn shrink_single_no_move_3() -> Result<()> {
|
||||||
|
let mut s = SingleThinS::new(1024, 1024, 2048, 2048);
|
||||||
|
test_shrink(&mut s)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn shrink_single_partial_move() -> Result<()> {
|
||||||
|
let mut s = SingleThinS::new(1024, 1024, 2048, 1280);
|
||||||
|
test_shrink(&mut s)
|
||||||
|
}
|
||||||
|
|
||||||
|
//------------------------------------
|
||||||
|
Loading…
Reference in New Issue
Block a user