Skip to main content

btrfs_cli/filesystem/
usage.rs

1use 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/// Show detailed information about internal filesystem usage
22#[derive(Parser, Debug)]
23pub struct FilesystemUsageCommand {
24    #[clap(flatten)]
25    pub units: UnitMode,
26
27    /// Use base 1000 for human-readable sizes
28    #[clap(short = 'H', long)]
29    pub human_si: bool,
30
31    /// Show data in tabular format
32    #[clap(short = 'T', long)]
33    pub tabular: bool,
34
35    /// One or more mount points to show usage for
36    #[clap(required = true)]
37    pub paths: Vec<PathBuf>,
38}
39
40/// Number of raw-device copies per chunk for a given profile.
41/// Returns 0 for RAID5/6 which require chunk tree data to compute accurately.
42fn 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    // Per-device chunk allocations from the chunk tree.  This requires
123    // CAP_SYS_ADMIN; if it fails we degrade gracefully and note it below.
124    let chunk_allocs = device_chunk_allocations(fd).ok();
125
126    // Map devid -> path for display.
127    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        // Per-device lines: one row per device that holds stripes for this
276        // exact profile.  Sorted by devid for stable output.
277        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}