1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
use binrw::{io::SeekFrom, BinRead, BinReaderExt};
use log::debug;
use simple_error::bail;
use std::{
error::Error,
fs::File,
io::BufReader,
path::Path,
process::{Command, Stdio},
};
mod header;
pub use header::*;
pub mod levels;
use levels::*;
#[derive(BinRead, Debug)]
#[brw(big)]
pub struct Qcow3 {
pub header: QcowHeader,
#[br(seek_before = SeekFrom::Start(header.l1_table_offset), count = header.l1_size)]
pub l1_table: Vec<L1Entry>,
#[br(ignore)]
pub path: String,
}
impl Qcow3 {
pub fn open(path: impl AsRef<Path>) -> Result<Self, Box<dyn Error>> {
let mut file = BufReader::new(File::open(&path)?);
let mut qcow: Qcow3 = file.read_be()?;
qcow.path = path.as_ref().to_string_lossy().to_string();
debug!("Opened qcow image: {:?}", &qcow);
Ok(qcow)
}
pub fn create(path: &str, size: u64) -> Result<Self, Box<dyn Error>> {
let status = Command::new("qemu-img")
.args([
"create",
"-f",
"qcow2",
"-o",
"cluster_size=65536",
&path,
&format!("{size}"),
])
.stdout(Stdio::null())
.stderr(Stdio::null())
.status()
.unwrap();
if status.code().unwrap() != 0 {
bail!("Failed to allocate image with qemu-img");
}
Qcow3::open(path)
}
pub fn count_clusters(&self) -> Result<u64, Box<dyn Error>> {
let mut count = 0;
for l1_entry in &self.l1_table {
if let Some(l2_table) =
l1_entry.read_l2(&mut File::open(&self.path)?, self.header.cluster_bits)
{
for l2_entry in l2_table {
if l2_entry.is_used {
count += 1;
}
}
}
}
Ok(count)
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_open() -> Result<(), Box<dyn Error>> {
let qcow = Qcow3::open("test/empty.qcow2")?;
assert_eq!(qcow.header.cluster_bits, 16);
assert_eq!(qcow.header.cluster_size(), 65536);
Ok(())
}
}