Introduce fstat() and adopt it in reader

Begin to migrate to Rust "native" Metadata, as part of addressing #10634

This will be structured as a series of small commits to aid bisecting.
This commit is contained in:
Peter Ammon 2024-07-27 11:39:25 -07:00
parent 0651ca0d9b
commit 1332d33025
No known key found for this signature in database
2 changed files with 24 additions and 13 deletions

View file

@ -15,7 +15,7 @@
use libc::{ use libc::{
c_char, c_int, ECHO, EINTR, EIO, EISDIR, ENOTTY, EPERM, ESRCH, ICANON, ICRNL, IEXTEN, INLCR, c_char, c_int, ECHO, EINTR, EIO, EISDIR, ENOTTY, EPERM, ESRCH, ICANON, ICRNL, IEXTEN, INLCR,
IXOFF, IXON, ONLCR, OPOST, O_NONBLOCK, O_RDONLY, SIGINT, SIGTTIN, STDIN_FILENO, STDOUT_FILENO, IXOFF, IXON, ONLCR, OPOST, O_NONBLOCK, O_RDONLY, SIGINT, SIGTTIN, STDIN_FILENO, STDOUT_FILENO,
S_IFDIR, TCSANOW, VMIN, VQUIT, VSUSP, VTIME, _POSIX_VDISABLE, TCSANOW, VMIN, VQUIT, VSUSP, VTIME, _POSIX_VDISABLE,
}; };
use nix::fcntl::OFlag; use nix::fcntl::OFlag;
use nix::sys::stat::Mode; use nix::sys::stat::Mode;
@ -120,7 +120,7 @@ use crate::wcstringutil::{
string_prefixes_string_case_insensitive, StringFuzzyMatch, string_prefixes_string_case_insensitive, StringFuzzyMatch,
}; };
use crate::wildcard::wildcard_has; use crate::wildcard::wildcard_has;
use crate::wutil::{perror, write_to_fd}; use crate::wutil::{fstat, perror, write_to_fd};
use crate::{abbrs, event, function, history}; use crate::{abbrs, event, function, history};
/// A description of where fish is in the process of exiting. /// A description of where fish is in the process of exiting.
@ -699,20 +699,21 @@ fn read_i(parser: &Parser) -> i32 {
/// highlighting. This is used for reading scripts and init files. /// highlighting. This is used for reading scripts and init files.
/// The file is not closed. /// The file is not closed.
fn read_ni(parser: &Parser, fd: RawFd, io: &IoChain) -> i32 { fn read_ni(parser: &Parser, fd: RawFd, io: &IoChain) -> i32 {
let mut buf: libc::stat = unsafe { std::mem::zeroed() }; let md = match fstat(fd) {
if unsafe { libc::fstat(fd, &mut buf) } == -1 { Ok(md) => md,
let err = errno(); Err(err) => {
FLOG!( FLOG!(
error, error,
wgettext_fmt!("Unable to read input file: %s", err.to_string()) wgettext_fmt!("Unable to read input file: %s", err.to_string())
); );
return 1; return 1;
} }
};
/* FreeBSD allows read() on directories. Error explicitly in that case. */ /* FreeBSD allows read() on directories. Error explicitly in that case. */
// XXX: This can be triggered spuriously, so we'll not do that for stdin. // XXX: This can be triggered spuriously, so we'll not do that for stdin.
// This can be seen e.g. with node's "spawn" api. // This can be seen e.g. with node's "spawn" api.
if fd != STDIN_FILENO && (buf.st_mode & S_IFDIR) != 0 { if fd != STDIN_FILENO && md.is_dir() {
FLOG!( FLOG!(
error, error,
wgettext_fmt!("Unable to read input file: %s", Errno(EISDIR).to_string()) wgettext_fmt!("Unable to read input file: %s", Errno(EISDIR).to_string())
@ -721,7 +722,7 @@ fn read_ni(parser: &Parser, fd: RawFd, io: &IoChain) -> i32 {
} }
// Read all data into a vec. // Read all data into a vec.
let mut fd_contents = Vec::with_capacity(usize::try_from(buf.st_size).unwrap()); let mut fd_contents = Vec::with_capacity(usize::try_from(md.len()).unwrap());
loop { loop {
let mut buff = [0_u8; 4096]; let mut buff = [0_u8; 4096];

View file

@ -52,6 +52,16 @@ pub fn lwstat(file_name: &wstr) -> io::Result<fs::Metadata> {
fs::symlink_metadata(tmp) fs::symlink_metadata(tmp)
} }
/// Cover over fstat().
pub fn fstat(fd: impl AsRawFd) -> io::Result<fs::Metadata> {
let fd = fd.as_raw_fd();
let file = unsafe { fs::File::from_raw_fd(fd) };
let res = file.metadata();
let fd2 = file.into_raw_fd();
assert_eq!(fd, fd2);
res
}
/// Wide character version of access(). /// Wide character version of access().
pub fn waccess(file_name: &wstr, mode: libc::c_int) -> libc::c_int { pub fn waccess(file_name: &wstr, mode: libc::c_int) -> libc::c_int {
let tmp = wcs2zstring(file_name); let tmp = wcs2zstring(file_name);