1#[cfg(not(linux_android))]
5use std::ffi::CStr;
6use std::fmt::{self, Debug};
7use std::mem;
8use std::os::unix::io::{AsFd, AsRawFd};
9
10use cfg_if::cfg_if;
11
12#[cfg(all(feature = "mount", bsd))]
13use crate::mount::MntFlags;
14#[cfg(target_os = "linux")]
15use crate::sys::statvfs::FsFlags;
16use crate::{errno::Errno, NixPath, Result};
17
18#[cfg(target_os = "android")]
20pub type fsid_t = libc::__fsid_t;
21#[cfg(not(target_os = "android"))]
23pub type fsid_t = libc::fsid_t;
24
25cfg_if! {
26 if #[cfg(any(linux_android, target_os = "fuchsia"))] {
27 type type_of_statfs = libc::statfs64;
28 const LIBC_FSTATFS: unsafe extern fn
29 (fd: libc::c_int, buf: *mut type_of_statfs) -> libc::c_int
30 = libc::fstatfs64;
31 const LIBC_STATFS: unsafe extern fn
32 (path: *const libc::c_char, buf: *mut type_of_statfs) -> libc::c_int
33 = libc::statfs64;
34 } else {
35 type type_of_statfs = libc::statfs;
36 const LIBC_FSTATFS: unsafe extern fn
37 (fd: libc::c_int, buf: *mut type_of_statfs) -> libc::c_int
38 = libc::fstatfs;
39 const LIBC_STATFS: unsafe extern fn
40 (path: *const libc::c_char, buf: *mut type_of_statfs) -> libc::c_int
41 = libc::statfs;
42 }
43}
44
45#[derive(Clone, Copy)]
47#[repr(transparent)]
48pub struct Statfs(type_of_statfs);
49
50#[cfg(target_os = "freebsd")]
51type fs_type_t = u32;
52#[cfg(target_os = "android")]
53type fs_type_t = libc::c_ulong;
54#[cfg(all(target_os = "linux", target_arch = "s390x", not(target_env = "musl")))]
55type fs_type_t = libc::c_uint;
56#[cfg(all(target_os = "linux", target_env = "musl"))]
57type fs_type_t = libc::c_ulong;
58#[cfg(all(target_os = "linux", target_env = "ohos"))]
59type fs_type_t = libc::c_ulong;
60#[cfg(all(target_os = "linux", target_env = "uclibc"))]
61type fs_type_t = libc::c_int;
62#[cfg(all(
63 target_os = "linux",
64 not(any(
65 target_arch = "s390x",
66 target_env = "musl",
67 target_env = "ohos",
68 target_env = "uclibc"
69 ))
70))]
71type fs_type_t = libc::__fsword_t;
72
73#[cfg(any(
75 target_os = "freebsd",
76 target_os = "android",
77 all(target_os = "linux", target_arch = "s390x"),
78 all(target_os = "linux", target_env = "musl"),
79 all(target_os = "linux", target_env = "ohos"),
80 all(
81 target_os = "linux",
82 not(any(target_arch = "s390x", target_env = "musl"))
83 ),
84))]
85#[derive(Eq, Copy, Clone, PartialEq, Debug)]
86pub struct FsType(pub fs_type_t);
87
88#[cfg(linux_android)]
91#[allow(missing_docs)]
92pub const ADFS_SUPER_MAGIC: FsType =
93 FsType(libc::ADFS_SUPER_MAGIC as fs_type_t);
94#[cfg(linux_android)]
95#[allow(missing_docs)]
96pub const AFFS_SUPER_MAGIC: FsType =
97 FsType(libc::AFFS_SUPER_MAGIC as fs_type_t);
98#[cfg(linux_android)]
99#[allow(missing_docs)]
100pub const AFS_SUPER_MAGIC: FsType = FsType(libc::AFS_SUPER_MAGIC as fs_type_t);
101#[cfg(linux_android)]
102#[allow(missing_docs)]
103pub const AUTOFS_SUPER_MAGIC: FsType =
104 FsType(libc::AUTOFS_SUPER_MAGIC as fs_type_t);
105#[cfg(linux_android)]
106#[allow(missing_docs)]
107pub const BPF_FS_MAGIC: FsType = FsType(libc::BPF_FS_MAGIC as fs_type_t);
108#[cfg(linux_android)]
109#[allow(missing_docs)]
110pub const BTRFS_SUPER_MAGIC: FsType =
111 FsType(libc::BTRFS_SUPER_MAGIC as fs_type_t);
112#[cfg(linux_android)]
113#[allow(missing_docs)]
114pub const CGROUP2_SUPER_MAGIC: FsType =
115 FsType(libc::CGROUP2_SUPER_MAGIC as fs_type_t);
116#[cfg(linux_android)]
117#[allow(missing_docs)]
118pub const CGROUP_SUPER_MAGIC: FsType =
119 FsType(libc::CGROUP_SUPER_MAGIC as fs_type_t);
120#[cfg(linux_android)]
121#[allow(missing_docs)]
122pub const CODA_SUPER_MAGIC: FsType =
123 FsType(libc::CODA_SUPER_MAGIC as fs_type_t);
124#[cfg(linux_android)]
125#[allow(missing_docs)]
126pub const CRAMFS_MAGIC: FsType = FsType(libc::CRAMFS_MAGIC as fs_type_t);
127#[cfg(linux_android)]
128#[allow(missing_docs)]
129pub const DEBUGFS_MAGIC: FsType = FsType(libc::DEBUGFS_MAGIC as fs_type_t);
130#[cfg(linux_android)]
131#[allow(missing_docs)]
132pub const DEVPTS_SUPER_MAGIC: FsType =
133 FsType(libc::DEVPTS_SUPER_MAGIC as fs_type_t);
134#[cfg(linux_android)]
135#[allow(missing_docs)]
136pub const ECRYPTFS_SUPER_MAGIC: FsType =
137 FsType(libc::ECRYPTFS_SUPER_MAGIC as fs_type_t);
138#[cfg(linux_android)]
139#[allow(missing_docs)]
140pub const EFS_SUPER_MAGIC: FsType = FsType(libc::EFS_SUPER_MAGIC as fs_type_t);
141#[cfg(linux_android)]
142#[allow(missing_docs)]
143pub const EXT2_SUPER_MAGIC: FsType =
144 FsType(libc::EXT2_SUPER_MAGIC as fs_type_t);
145#[cfg(linux_android)]
146#[allow(missing_docs)]
147pub const EXT3_SUPER_MAGIC: FsType =
148 FsType(libc::EXT3_SUPER_MAGIC as fs_type_t);
149#[cfg(linux_android)]
150#[allow(missing_docs)]
151pub const EXT4_SUPER_MAGIC: FsType =
152 FsType(libc::EXT4_SUPER_MAGIC as fs_type_t);
153#[cfg(linux_android)]
154#[allow(missing_docs)]
155pub const F2FS_SUPER_MAGIC: FsType =
156 FsType(libc::F2FS_SUPER_MAGIC as fs_type_t);
157#[cfg(linux_android)]
158#[allow(missing_docs)]
159pub const FUSE_SUPER_MAGIC: FsType =
160 FsType(libc::FUSE_SUPER_MAGIC as fs_type_t);
161#[cfg(linux_android)]
162#[allow(missing_docs)]
163pub const FUTEXFS_SUPER_MAGIC: FsType =
164 FsType(libc::FUTEXFS_SUPER_MAGIC as fs_type_t);
165#[cfg(linux_android)]
166#[allow(missing_docs)]
167pub const HOSTFS_SUPER_MAGIC: FsType =
168 FsType(libc::HOSTFS_SUPER_MAGIC as fs_type_t);
169#[cfg(linux_android)]
170#[allow(missing_docs)]
171pub const HPFS_SUPER_MAGIC: FsType =
172 FsType(libc::HPFS_SUPER_MAGIC as fs_type_t);
173#[cfg(linux_android)]
174#[allow(missing_docs)]
175pub const HUGETLBFS_MAGIC: FsType = FsType(libc::HUGETLBFS_MAGIC as fs_type_t);
176#[cfg(linux_android)]
177#[allow(missing_docs)]
178pub const ISOFS_SUPER_MAGIC: FsType =
179 FsType(libc::ISOFS_SUPER_MAGIC as fs_type_t);
180#[cfg(linux_android)]
181#[allow(missing_docs)]
182pub const JFFS2_SUPER_MAGIC: FsType =
183 FsType(libc::JFFS2_SUPER_MAGIC as fs_type_t);
184#[cfg(linux_android)]
185#[allow(missing_docs)]
186pub const MINIX2_SUPER_MAGIC2: FsType =
187 FsType(libc::MINIX2_SUPER_MAGIC2 as fs_type_t);
188#[cfg(linux_android)]
189#[allow(missing_docs)]
190pub const MINIX2_SUPER_MAGIC: FsType =
191 FsType(libc::MINIX2_SUPER_MAGIC as fs_type_t);
192#[cfg(linux_android)]
193#[allow(missing_docs)]
194pub const MINIX3_SUPER_MAGIC: FsType =
195 FsType(libc::MINIX3_SUPER_MAGIC as fs_type_t);
196#[cfg(linux_android)]
197#[allow(missing_docs)]
198pub const MINIX_SUPER_MAGIC2: FsType =
199 FsType(libc::MINIX_SUPER_MAGIC2 as fs_type_t);
200#[cfg(linux_android)]
201#[allow(missing_docs)]
202pub const MINIX_SUPER_MAGIC: FsType =
203 FsType(libc::MINIX_SUPER_MAGIC as fs_type_t);
204#[cfg(linux_android)]
205#[allow(missing_docs)]
206pub const MSDOS_SUPER_MAGIC: FsType =
207 FsType(libc::MSDOS_SUPER_MAGIC as fs_type_t);
208#[cfg(linux_android)]
209#[allow(missing_docs)]
210pub const NCP_SUPER_MAGIC: FsType = FsType(libc::NCP_SUPER_MAGIC as fs_type_t);
211#[cfg(linux_android)]
212#[allow(missing_docs)]
213pub const NFS_SUPER_MAGIC: FsType = FsType(libc::NFS_SUPER_MAGIC as fs_type_t);
214#[cfg(linux_android)]
215#[allow(missing_docs)]
216pub const NILFS_SUPER_MAGIC: FsType =
217 FsType(libc::NILFS_SUPER_MAGIC as fs_type_t);
218#[cfg(linux_android)]
219#[allow(missing_docs)]
220pub const OCFS2_SUPER_MAGIC: FsType =
221 FsType(libc::OCFS2_SUPER_MAGIC as fs_type_t);
222#[cfg(linux_android)]
223#[allow(missing_docs)]
224pub const OPENPROM_SUPER_MAGIC: FsType =
225 FsType(libc::OPENPROM_SUPER_MAGIC as fs_type_t);
226#[cfg(linux_android)]
227#[allow(missing_docs)]
228pub const OVERLAYFS_SUPER_MAGIC: FsType =
229 FsType(libc::OVERLAYFS_SUPER_MAGIC as fs_type_t);
230#[cfg(linux_android)]
231#[allow(missing_docs)]
232pub const PROC_SUPER_MAGIC: FsType =
233 FsType(libc::PROC_SUPER_MAGIC as fs_type_t);
234#[cfg(linux_android)]
235#[allow(missing_docs)]
236pub const QNX4_SUPER_MAGIC: FsType =
237 FsType(libc::QNX4_SUPER_MAGIC as fs_type_t);
238#[cfg(linux_android)]
239#[allow(missing_docs)]
240pub const QNX6_SUPER_MAGIC: FsType =
241 FsType(libc::QNX6_SUPER_MAGIC as fs_type_t);
242#[cfg(linux_android)]
243#[allow(missing_docs)]
244pub const RDTGROUP_SUPER_MAGIC: FsType =
245 FsType(libc::RDTGROUP_SUPER_MAGIC as fs_type_t);
246#[cfg(linux_android)]
247#[allow(missing_docs)]
248pub const REISERFS_SUPER_MAGIC: FsType =
249 FsType(libc::REISERFS_SUPER_MAGIC as fs_type_t);
250#[cfg(linux_android)]
251#[allow(missing_docs)]
252pub const SECURITYFS_MAGIC: FsType =
253 FsType(libc::SECURITYFS_MAGIC as fs_type_t);
254#[cfg(linux_android)]
255#[allow(missing_docs)]
256pub const SELINUX_MAGIC: FsType = FsType(libc::SELINUX_MAGIC as fs_type_t);
257#[cfg(linux_android)]
258#[allow(missing_docs)]
259pub const SMACK_MAGIC: FsType = FsType(libc::SMACK_MAGIC as fs_type_t);
260#[cfg(linux_android)]
261#[allow(missing_docs)]
262pub const SMB_SUPER_MAGIC: FsType = FsType(libc::SMB_SUPER_MAGIC as fs_type_t);
263#[cfg(linux_android)]
264#[allow(missing_docs)]
265pub const SYSFS_MAGIC: FsType = FsType(libc::SYSFS_MAGIC as fs_type_t);
266#[cfg(linux_android)]
267#[allow(missing_docs)]
268pub const TMPFS_MAGIC: FsType = FsType(libc::TMPFS_MAGIC as fs_type_t);
269#[cfg(linux_android)]
270#[allow(missing_docs)]
271pub const TRACEFS_MAGIC: FsType = FsType(libc::TRACEFS_MAGIC as fs_type_t);
272#[cfg(linux_android)]
273#[allow(missing_docs)]
274pub const UDF_SUPER_MAGIC: FsType = FsType(libc::UDF_SUPER_MAGIC as fs_type_t);
275#[cfg(linux_android)]
276#[allow(missing_docs)]
277pub const USBDEVICE_SUPER_MAGIC: FsType =
278 FsType(libc::USBDEVICE_SUPER_MAGIC as fs_type_t);
279#[cfg(linux_android)]
280#[allow(missing_docs)]
281pub const XENFS_SUPER_MAGIC: FsType =
282 FsType(libc::XENFS_SUPER_MAGIC as fs_type_t);
283#[cfg(linux_android)]
284#[allow(missing_docs)]
285pub const NSFS_MAGIC: FsType = FsType(libc::NSFS_MAGIC as fs_type_t);
286#[cfg(all(linux_android, not(target_env = "musl"), not(target_env = "ohos")))]
287#[allow(missing_docs)]
288pub const XFS_SUPER_MAGIC: FsType = FsType(libc::XFS_SUPER_MAGIC as fs_type_t);
289
290impl Statfs {
291 #[cfg(not(any(
293 target_os = "openbsd",
294 target_os = "dragonfly",
295 apple_targets,
296 )))]
297 pub fn filesystem_type(&self) -> FsType {
298 FsType(self.0.f_type)
299 }
300
301 #[cfg(not(linux_android))]
303 pub fn filesystem_type_name(&self) -> &str {
304 let c_str = unsafe { CStr::from_ptr(self.0.f_fstypename.as_ptr()) };
305 c_str.to_str().unwrap()
306 }
307
308 #[cfg(apple_targets)]
310 pub fn optimal_transfer_size(&self) -> i32 {
311 self.0.f_iosize
312 }
313
314 #[cfg(target_os = "openbsd")]
316 pub fn optimal_transfer_size(&self) -> u32 {
317 self.0.f_iosize
318 }
319
320 #[cfg(all(target_os = "linux", target_arch = "s390x", not(target_env = "musl")))]
322 pub fn optimal_transfer_size(&self) -> u32 {
323 self.0.f_bsize
324 }
325
326 #[cfg(any(
328 target_os = "android",
329 all(target_os = "linux", target_env = "musl"),
330 all(target_os = "linux", target_env = "ohos")
331 ))]
332 pub fn optimal_transfer_size(&self) -> libc::c_ulong {
333 self.0.f_bsize
334 }
335
336 #[cfg(all(
338 target_os = "linux",
339 not(any(
340 target_arch = "s390x",
341 target_env = "musl",
342 target_env = "ohos",
343 target_env = "uclibc"
344 ))
345 ))]
346 pub fn optimal_transfer_size(&self) -> libc::__fsword_t {
347 self.0.f_bsize
348 }
349
350 #[cfg(all(target_os = "linux", target_env = "uclibc"))]
352 pub fn optimal_transfer_size(&self) -> libc::c_int {
353 self.0.f_bsize
354 }
355
356 #[cfg(target_os = "dragonfly")]
358 pub fn optimal_transfer_size(&self) -> libc::c_long {
359 self.0.f_iosize
360 }
361
362 #[cfg(target_os = "freebsd")]
364 pub fn optimal_transfer_size(&self) -> u64 {
365 self.0.f_iosize
366 }
367
368 #[cfg(any(apple_targets, target_os = "openbsd"))]
370 pub fn block_size(&self) -> u32 {
371 self.0.f_bsize
372 }
373
374 #[cfg(all(target_os = "linux", target_arch = "s390x", not(target_env = "musl")))]
377 pub fn block_size(&self) -> u32 {
378 self.0.f_bsize
379 }
380
381 #[cfg(all(target_os = "linux", target_env = "musl"))]
384 pub fn block_size(&self) -> libc::c_ulong {
385 self.0.f_bsize
386 }
387
388 #[cfg(all(target_os = "linux", target_env = "ohos"))]
391 pub fn block_size(&self) -> libc::c_ulong {
392 self.0.f_bsize
393 }
394
395 #[cfg(all(target_os = "linux", target_env = "uclibc"))]
398 pub fn block_size(&self) -> libc::c_int {
399 self.0.f_bsize
400 }
401
402 #[cfg(all(
405 target_os = "linux",
406 not(any(
407 target_arch = "s390x",
408 target_env = "musl",
409 target_env = "ohos",
410 target_env = "uclibc"
411 ))
412 ))]
413 pub fn block_size(&self) -> libc::__fsword_t {
414 self.0.f_bsize
415 }
416
417 #[cfg(target_os = "freebsd")]
419 pub fn block_size(&self) -> u64 {
420 self.0.f_bsize
421 }
422
423 #[cfg(target_os = "android")]
425 pub fn block_size(&self) -> libc::c_ulong {
426 self.0.f_bsize
427 }
428
429 #[cfg(target_os = "dragonfly")]
431 pub fn block_size(&self) -> libc::c_long {
432 self.0.f_bsize
433 }
434
435 #[cfg(all(feature = "mount", bsd))]
437 #[allow(clippy::unnecessary_cast)] pub fn flags(&self) -> MntFlags {
439 MntFlags::from_bits_truncate(self.0.f_flags as i32)
440 }
441
442 #[cfg(target_os = "linux")]
446 pub fn flags(&self) -> FsFlags {
447 FsFlags::from_bits_truncate(self.0.f_flags as libc::c_ulong)
448 }
449
450 #[cfg(any(target_os = "freebsd", target_os = "openbsd"))]
452 pub fn maximum_name_length(&self) -> u32 {
453 self.0.f_namemax
454 }
455
456 #[cfg(all(target_os = "linux", target_arch = "s390x", not(target_env = "musl")))]
458 pub fn maximum_name_length(&self) -> u32 {
459 self.0.f_namelen
460 }
461
462 #[cfg(all(target_os = "linux", target_env = "musl"))]
464 pub fn maximum_name_length(&self) -> libc::c_ulong {
465 self.0.f_namelen
466 }
467
468 #[cfg(all(target_os = "linux", target_env = "uclibc"))]
470 pub fn maximum_name_length(&self) -> libc::c_int {
471 self.0.f_namelen
472 }
473
474 #[cfg(all(
476 target_os = "linux",
477 not(any(
478 target_arch = "s390x",
479 target_env = "musl",
480 target_env = "ohos",
481 target_env = "uclibc"
482 ))
483 ))]
484 pub fn maximum_name_length(&self) -> libc::__fsword_t {
485 self.0.f_namelen
486 }
487
488 #[cfg(target_os = "android")]
490 pub fn maximum_name_length(&self) -> libc::c_ulong {
491 self.0.f_namelen
492 }
493
494 #[cfg(any(
496 apple_targets,
497 linux_android,
498 target_os = "freebsd",
499 target_os = "fuchsia",
500 target_os = "openbsd",
501 ))]
502 pub fn blocks(&self) -> u64 {
503 self.0.f_blocks
504 }
505
506 #[cfg(target_os = "dragonfly")]
508 pub fn blocks(&self) -> libc::c_long {
509 self.0.f_blocks
510 }
511
512 #[cfg(target_os = "emscripten")]
514 pub fn blocks(&self) -> u32 {
515 self.0.f_blocks
516 }
517
518 #[cfg(any(
520 apple_targets,
521 linux_android,
522 target_os = "freebsd",
523 target_os = "fuchsia",
524 target_os = "openbsd",
525 ))]
526 pub fn blocks_free(&self) -> u64 {
527 self.0.f_bfree
528 }
529
530 #[cfg(target_os = "dragonfly")]
532 pub fn blocks_free(&self) -> libc::c_long {
533 self.0.f_bfree
534 }
535
536 #[cfg(target_os = "emscripten")]
538 pub fn blocks_free(&self) -> u32 {
539 self.0.f_bfree
540 }
541
542 #[cfg(any(apple_targets, linux_android, target_os = "fuchsia"))]
544 pub fn blocks_available(&self) -> u64 {
545 self.0.f_bavail
546 }
547
548 #[cfg(target_os = "dragonfly")]
550 pub fn blocks_available(&self) -> libc::c_long {
551 self.0.f_bavail
552 }
553
554 #[cfg(any(target_os = "freebsd", target_os = "openbsd"))]
556 pub fn blocks_available(&self) -> i64 {
557 self.0.f_bavail
558 }
559
560 #[cfg(target_os = "emscripten")]
562 pub fn blocks_available(&self) -> u32 {
563 self.0.f_bavail
564 }
565
566 #[cfg(any(
568 apple_targets,
569 linux_android,
570 target_os = "freebsd",
571 target_os = "fuchsia",
572 target_os = "openbsd",
573 ))]
574 pub fn files(&self) -> u64 {
575 self.0.f_files
576 }
577
578 #[cfg(target_os = "dragonfly")]
580 pub fn files(&self) -> libc::c_long {
581 self.0.f_files
582 }
583
584 #[cfg(target_os = "emscripten")]
586 pub fn files(&self) -> u32 {
587 self.0.f_files
588 }
589
590 #[cfg(any(
592 apple_targets,
593 linux_android,
594 target_os = "fuchsia",
595 target_os = "openbsd",
596 ))]
597 pub fn files_free(&self) -> u64 {
598 self.0.f_ffree
599 }
600
601 #[cfg(target_os = "dragonfly")]
603 pub fn files_free(&self) -> libc::c_long {
604 self.0.f_ffree
605 }
606
607 #[cfg(target_os = "freebsd")]
609 pub fn files_free(&self) -> i64 {
610 self.0.f_ffree
611 }
612
613 #[cfg(target_os = "emscripten")]
615 pub fn files_free(&self) -> u32 {
616 self.0.f_ffree
617 }
618
619 pub fn filesystem_id(&self) -> fsid_t {
621 self.0.f_fsid
622 }
623}
624
625impl Debug for Statfs {
626 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
627 let mut ds = f.debug_struct("Statfs");
628 ds.field("optimal_transfer_size", &self.optimal_transfer_size());
629 ds.field("block_size", &self.block_size());
630 ds.field("blocks", &self.blocks());
631 ds.field("blocks_free", &self.blocks_free());
632 ds.field("blocks_available", &self.blocks_available());
633 ds.field("files", &self.files());
634 ds.field("files_free", &self.files_free());
635 ds.field("filesystem_id", &self.filesystem_id());
636 #[cfg(all(feature = "mount", bsd))]
637 ds.field("flags", &self.flags());
638 ds.finish()
639 }
640}
641
642pub fn statfs<P: ?Sized + NixPath>(path: &P) -> Result<Statfs> {
651 unsafe {
652 let mut stat = mem::MaybeUninit::<type_of_statfs>::uninit();
653 let res = path.with_nix_path(|path| {
654 LIBC_STATFS(path.as_ptr(), stat.as_mut_ptr())
655 })?;
656 Errno::result(res).map(|_| Statfs(stat.assume_init()))
657 }
658}
659
660pub fn fstatfs<Fd: AsFd>(fd: Fd) -> Result<Statfs> {
669 unsafe {
670 let mut stat = mem::MaybeUninit::<type_of_statfs>::uninit();
671 Errno::result(LIBC_FSTATFS(fd.as_fd().as_raw_fd(), stat.as_mut_ptr()))
672 .map(|_| Statfs(stat.assume_init()))
673 }
674}