btrfs_cli/filesystem/
usage.rs1use super::UnitMode;
2use crate::{
3 Format, Runnable,
4 util::{SizeFormat, fmt_size},
5};
6use anyhow::{Context, Result};
7use btrfs_uapi::{
8 chunk::device_chunk_allocations,
9 device::device_info_all,
10 filesystem::filesystem_info,
11 space::{BlockGroupFlags, SpaceInfo, space_info},
12};
13use clap::Parser;
14use std::{
15 collections::{HashMap, HashSet},
16 fs::File,
17 os::unix::io::AsFd,
18 path::PathBuf,
19};
20
21#[derive(Parser, Debug)]
23pub struct FilesystemUsageCommand {
24 #[clap(flatten)]
25 pub units: UnitMode,
26
27 #[clap(short = 'H', long)]
29 pub human_si: bool,
30
31 #[clap(short = 'T', long)]
33 pub tabular: bool,
34
35 #[clap(required = true)]
37 pub paths: Vec<PathBuf>,
38}
39
40fn profile_ncopies(flags: BlockGroupFlags) -> u64 {
43 if flags.contains(BlockGroupFlags::RAID1C4) {
44 4
45 } else if flags.contains(BlockGroupFlags::RAID1C3) {
46 3
47 } else if flags.contains(BlockGroupFlags::RAID1)
48 || flags.contains(BlockGroupFlags::DUP)
49 || flags.contains(BlockGroupFlags::RAID10)
50 {
51 2
52 } else {
53 u64::from(
54 !(flags.contains(BlockGroupFlags::RAID5)
55 || flags.contains(BlockGroupFlags::RAID6)),
56 )
57 }
58}
59
60fn has_multiple_profiles(spaces: &[SpaceInfo]) -> bool {
61 let profile_mask = BlockGroupFlags::RAID0
62 | BlockGroupFlags::RAID1
63 | BlockGroupFlags::DUP
64 | BlockGroupFlags::RAID10
65 | BlockGroupFlags::RAID5
66 | BlockGroupFlags::RAID6
67 | BlockGroupFlags::RAID1C3
68 | BlockGroupFlags::RAID1C4
69 | BlockGroupFlags::SINGLE;
70
71 let profiles_for = |type_flag: BlockGroupFlags| {
72 spaces
73 .iter()
74 .filter(|s| {
75 s.flags.contains(type_flag)
76 && !s.flags.contains(BlockGroupFlags::GLOBAL_RSV)
77 })
78 .map(|s| s.flags & profile_mask)
79 .collect::<HashSet<_>>()
80 };
81
82 profiles_for(BlockGroupFlags::DATA).len() > 1
83 || profiles_for(BlockGroupFlags::METADATA).len() > 1
84 || profiles_for(BlockGroupFlags::SYSTEM).len() > 1
85}
86
87impl Runnable for FilesystemUsageCommand {
88 fn run(&self, _format: Format, _dry_run: bool) -> Result<()> {
89 let mut mode = self.units.resolve();
90 if self.human_si {
91 mode = SizeFormat::HumanSi;
92 }
93 for (i, path) in self.paths.iter().enumerate() {
94 if i > 0 {
95 println!();
96 }
97 print_usage(path, self.tabular, &mode)?;
98 }
99 Ok(())
100 }
101}
102
103fn print_usage(
104 path: &std::path::Path,
105 _tabular: bool,
106 mode: &SizeFormat,
107) -> Result<()> {
108 let file = File::open(path)
109 .with_context(|| format!("failed to open '{}'", path.display()))?;
110 let fd = file.as_fd();
111
112 let fs = filesystem_info(fd).with_context(|| {
113 format!("failed to get filesystem info for '{}'", path.display())
114 })?;
115 let devices = device_info_all(fd, &fs).with_context(|| {
116 format!("failed to get device info for '{}'", path.display())
117 })?;
118 let spaces = space_info(fd).with_context(|| {
119 format!("failed to get space info for '{}'", path.display())
120 })?;
121
122 let chunk_allocs = device_chunk_allocations(fd).ok();
125
126 let devid_to_path: HashMap<u64, &str> =
128 devices.iter().map(|d| (d.devid, d.path.as_str())).collect();
129
130 let mut r_data_chunks: u64 = 0;
131 let mut r_data_used: u64 = 0;
132 let mut l_data_chunks: u64 = 0;
133 let mut r_meta_chunks: u64 = 0;
134 let mut r_meta_used: u64 = 0;
135 let mut l_meta_chunks: u64 = 0;
136 let mut r_sys_chunks: u64 = 0;
137 let mut r_sys_used: u64 = 0;
138 let mut l_global_reserve: u64 = 0;
139 let mut l_global_reserve_used: u64 = 0;
140 let mut max_ncopies: u64 = 1;
141
142 for s in &spaces {
143 if s.flags.contains(BlockGroupFlags::GLOBAL_RSV) {
144 l_global_reserve = s.total_bytes;
145 l_global_reserve_used = s.used_bytes;
146 continue;
147 }
148 let ncopies = profile_ncopies(s.flags);
149 if ncopies > max_ncopies {
150 max_ncopies = ncopies;
151 }
152 if s.flags.contains(BlockGroupFlags::DATA) {
153 r_data_chunks += s.total_bytes * ncopies;
154 r_data_used += s.used_bytes * ncopies;
155 l_data_chunks += s.total_bytes;
156 }
157 if s.flags.contains(BlockGroupFlags::METADATA) {
158 r_meta_chunks += s.total_bytes * ncopies;
159 r_meta_used += s.used_bytes * ncopies;
160 l_meta_chunks += s.total_bytes;
161 }
162 if s.flags.contains(BlockGroupFlags::SYSTEM) {
163 r_sys_chunks += s.total_bytes * ncopies;
164 r_sys_used += s.used_bytes * ncopies;
165 }
166 }
167
168 let r_total_size: u64 = devices.iter().map(|d| d.total_bytes).sum();
169 let r_total_chunks = r_data_chunks + r_meta_chunks + r_sys_chunks;
170 let r_total_used = r_data_used + r_meta_used + r_sys_used;
171 let r_total_unused = r_total_size.saturating_sub(r_total_chunks);
172
173 let r_total_missing: u64 = devices
174 .iter()
175 .filter(|d| std::fs::metadata(&d.path).is_err())
176 .map(|d| d.total_bytes)
177 .sum();
178
179 let data_ratio = if l_data_chunks > 0 {
180 r_data_chunks as f64 / l_data_chunks as f64
181 } else {
182 1.0
183 };
184 let meta_ratio = if l_meta_chunks > 0 {
185 r_meta_chunks as f64 / l_meta_chunks as f64
186 } else {
187 1.0
188 };
189 let max_data_ratio = max_ncopies as f64;
190
191 const MIN_UNALLOCATED_THRESH: u64 = 16 * 1024 * 1024;
192 let free_base = if data_ratio > 0.0 {
193 ((r_data_chunks.saturating_sub(r_data_used)) as f64 / data_ratio) as u64
194 } else {
195 0
196 };
197 let (free_estimated, free_min) = if r_total_unused >= MIN_UNALLOCATED_THRESH
198 {
199 (
200 free_base + (r_total_unused as f64 / data_ratio) as u64,
201 free_base + (r_total_unused as f64 / max_data_ratio) as u64,
202 )
203 } else {
204 (free_base, free_base)
205 };
206
207 let free_statfs = nix::sys::statfs::statfs(path)
208 .map(|st| st.blocks_available() * st.block_size() as u64)
209 .unwrap_or(0);
210
211 let multiple = has_multiple_profiles(&spaces);
212
213 println!("Overall:");
214 println!(" Device size:\t\t{:>10}", fmt_size(r_total_size, mode));
215 println!(
216 " Device allocated:\t\t{:>10}",
217 fmt_size(r_total_chunks, mode)
218 );
219 println!(
220 " Device unallocated:\t\t{:>10}",
221 fmt_size(r_total_unused, mode)
222 );
223 println!(
224 " Device missing:\t\t{:>10}",
225 fmt_size(r_total_missing, mode)
226 );
227 println!(" Device slack:\t\t{:>10}", fmt_size(0, mode));
228 println!(" Used:\t\t\t{:>10}", fmt_size(r_total_used, mode));
229 println!(
230 " Free (estimated):\t\t{:>10}\t(min: {})",
231 fmt_size(free_estimated, mode),
232 fmt_size(free_min, mode)
233 );
234 println!(
235 " Free (statfs, df):\t\t{:>10}",
236 fmt_size(free_statfs, mode)
237 );
238 println!(" Data ratio:\t\t\t{data_ratio:>10.2}");
239 println!(" Metadata ratio:\t\t{meta_ratio:>10.2}");
240 println!(
241 " Global reserve:\t\t{:>10}\t(used: {})",
242 fmt_size(l_global_reserve, mode),
243 fmt_size(l_global_reserve_used, mode)
244 );
245 println!(
246 " Multiple profiles:\t\t{:>10}",
247 if multiple { "yes" } else { "no" }
248 );
249
250 if chunk_allocs.is_none() {
251 eprintln!(
252 "NOTE: per-device usage breakdown unavailable \
253 (chunk tree requires CAP_SYS_ADMIN)"
254 );
255 }
256
257 for s in &spaces {
258 if s.flags.contains(BlockGroupFlags::GLOBAL_RSV) {
259 continue;
260 }
261 let pct = if s.total_bytes > 0 {
262 100.0 * s.used_bytes as f64 / s.total_bytes as f64
263 } else {
264 0.0
265 };
266 println!(
267 "\n{},{}: Size:{}, Used:{} ({:.2}%)",
268 s.flags.type_name(),
269 s.flags.profile_name(),
270 fmt_size(s.total_bytes, mode),
271 fmt_size(s.used_bytes, mode),
272 pct
273 );
274
275 if let Some(allocs) = &chunk_allocs {
278 let mut profile_allocs: Vec<_> =
279 allocs.iter().filter(|a| a.flags == s.flags).collect();
280 profile_allocs.sort_by_key(|a| a.devid);
281
282 for alloc in profile_allocs {
283 let path = devid_to_path
284 .get(&alloc.devid)
285 .copied()
286 .unwrap_or("<unknown>");
287 println!(" {}\t\t{:>10}", path, fmt_size(alloc.bytes, mode));
288 }
289 }
290 }
291
292 println!("\nUnallocated:");
293 for dev in &devices {
294 let unallocated = dev.total_bytes.saturating_sub(dev.bytes_used);
295 println!(" {}\t{:>10}", dev.path, fmt_size(unallocated, mode));
296 }
297
298 Ok(())
299}