2020-06-10 14:38:42 +01:00
|
|
|
use nix::sys::stat;
|
|
|
|
use nix::sys::stat::{FileStat, SFlag};
|
2020-06-29 10:49:40 +01:00
|
|
|
use std::fs::{File, OpenOptions};
|
2020-06-26 16:44:47 +01:00
|
|
|
use std::io;
|
|
|
|
use std::io::{Seek, Write};
|
2020-06-10 14:38:42 +01:00
|
|
|
use std::os::unix::io::AsRawFd;
|
2020-06-29 10:49:40 +01:00
|
|
|
use std::path::Path;
|
2020-06-26 16:44:47 +01:00
|
|
|
use tempfile::tempfile;
|
2020-06-10 14:38:42 +01:00
|
|
|
|
|
|
|
//---------------------------------------
|
|
|
|
|
|
|
|
fn check_bits(mode: u32, flag: &SFlag) -> bool {
|
|
|
|
(mode & flag.bits()) != 0
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_file_or_blk(info: FileStat) -> bool {
|
2020-06-26 16:44:47 +01:00
|
|
|
check_bits(info.st_mode, &stat::SFlag::S_IFBLK)
|
|
|
|
|| check_bits(info.st_mode, &stat::SFlag::S_IFREG)
|
2020-06-10 14:38:42 +01:00
|
|
|
}
|
2020-06-09 14:59:13 +01:00
|
|
|
|
2020-06-29 10:49:40 +01:00
|
|
|
pub fn file_exists(path: &Path) -> bool {
|
2020-06-10 14:38:42 +01:00
|
|
|
match stat::stat(path) {
|
2020-06-26 16:44:47 +01:00
|
|
|
Ok(info) => is_file_or_blk(info),
|
2020-06-10 14:38:42 +01:00
|
|
|
_ => {
|
|
|
|
// FIXME: assuming all errors indicate the file doesn't
|
|
|
|
// exist.
|
2020-06-10 16:46:38 +01:00
|
|
|
false
|
2020-06-10 14:38:42 +01:00
|
|
|
}
|
2020-06-09 14:59:13 +01:00
|
|
|
}
|
|
|
|
}
|
2020-06-10 14:38:42 +01:00
|
|
|
|
|
|
|
//---------------------------------------
|
|
|
|
|
|
|
|
const BLKGETSIZE64_CODE: u8 = 0x12;
|
|
|
|
const BLKGETSIZE64_SEQ: u8 = 114;
|
|
|
|
ioctl_read!(ioctl_blkgetsize64, BLKGETSIZE64_CODE, BLKGETSIZE64_SEQ, u64);
|
|
|
|
|
|
|
|
pub fn fail<T>(msg: &str) -> io::Result<T> {
|
|
|
|
let e = io::Error::new(io::ErrorKind::Other, msg);
|
|
|
|
Err(e)
|
|
|
|
}
|
|
|
|
|
2020-06-29 10:49:40 +01:00
|
|
|
fn get_device_size(path: &Path) -> io::Result<u64> {
|
2020-06-26 16:44:47 +01:00
|
|
|
let file = File::open(path)?;
|
2020-06-10 15:07:40 +01:00
|
|
|
let fd = file.as_raw_fd();
|
2020-06-10 14:38:42 +01:00
|
|
|
let mut cap = 0u64;
|
|
|
|
unsafe {
|
2020-06-26 16:44:47 +01:00
|
|
|
match ioctl_blkgetsize64(fd, &mut cap) {
|
|
|
|
Ok(_) => Ok(cap),
|
|
|
|
_ => fail("BLKGETSIZE64 ioctl failed"),
|
|
|
|
}
|
2020-06-10 14:38:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-29 10:49:40 +01:00
|
|
|
pub fn file_size(path: &Path) -> io::Result<u64> {
|
2020-06-10 14:38:42 +01:00
|
|
|
match stat::stat(path) {
|
|
|
|
Ok(info) => {
|
|
|
|
if check_bits(info.st_mode, &SFlag::S_IFREG) {
|
2020-06-10 16:46:38 +01:00
|
|
|
Ok(info.st_size as u64)
|
2020-06-10 14:38:42 +01:00
|
|
|
} else if check_bits(info.st_mode, &SFlag::S_IFBLK) {
|
2020-06-10 16:46:38 +01:00
|
|
|
get_device_size(path)
|
2020-06-10 14:38:42 +01:00
|
|
|
} else {
|
2020-06-10 16:46:38 +01:00
|
|
|
fail("not a regular file or block device")
|
2020-06-26 16:44:47 +01:00
|
|
|
}
|
2020-06-10 14:38:42 +01:00
|
|
|
}
|
2020-06-26 16:44:47 +01:00
|
|
|
_ => fail("stat failed"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------
|
|
|
|
|
2020-06-29 10:49:40 +01:00
|
|
|
fn set_size<W: Write + Seek>(w: &mut W, nr_bytes: u64) -> io::Result<()> {
|
2020-06-26 16:44:47 +01:00
|
|
|
let zeroes: Vec<u8> = vec![0; 1];
|
|
|
|
|
|
|
|
if nr_bytes > 0 {
|
2020-06-29 10:49:40 +01:00
|
|
|
w.seek(io::SeekFrom::Start(nr_bytes - 1))?;
|
|
|
|
w.write_all(&zeroes)?;
|
2020-06-26 16:44:47 +01:00
|
|
|
}
|
|
|
|
|
2020-06-29 10:49:40 +01:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn temp_file_sized(nr_bytes: u64) -> io::Result<std::fs::File> {
|
|
|
|
let mut file = tempfile()?;
|
|
|
|
set_size(&mut file, nr_bytes)?;
|
|
|
|
Ok(file)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_sized_file(path: &Path, nr_bytes: u64) -> io::Result<std::fs::File> {
|
|
|
|
let mut file = OpenOptions::new()
|
|
|
|
.read(false)
|
|
|
|
.write(true)
|
|
|
|
.create(true)
|
|
|
|
.truncate(true)
|
|
|
|
.open(path)?;
|
|
|
|
set_size(&mut file, nr_bytes)?;
|
2020-06-26 16:44:47 +01:00
|
|
|
Ok(file)
|
2020-06-10 14:38:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------
|