mirror of
https://github.com/asterinas/asterinas.git
synced 2025-06-25 10:23:23 +00:00
122 lines
2.8 KiB
Rust
122 lines
2.8 KiB
Rust
use core::time::Duration;
|
|
|
|
use crate::fs::utils::{FileSystem, Inode, InodeMode, InodeType, IoctlCmd, Metadata};
|
|
use crate::prelude::*;
|
|
|
|
use super::{ProcFS, ProcInodeInfo};
|
|
|
|
pub struct ProcFile<F: FileOps> {
|
|
inner: F,
|
|
info: ProcInodeInfo,
|
|
}
|
|
|
|
impl<F: FileOps> ProcFile<F> {
|
|
pub fn new(file: F, fs: Arc<dyn FileSystem>, is_volatile: bool) -> Arc<Self> {
|
|
let info = {
|
|
let procfs = fs.downcast_ref::<ProcFS>().unwrap();
|
|
let metadata = Metadata::new_file(
|
|
procfs.alloc_id(),
|
|
InodeMode::from_bits_truncate(0o444),
|
|
&fs.sb(),
|
|
);
|
|
ProcInodeInfo::new(metadata, Arc::downgrade(&fs), is_volatile)
|
|
};
|
|
Arc::new(Self { inner: file, info })
|
|
}
|
|
}
|
|
|
|
impl<F: FileOps + 'static> Inode for ProcFile<F> {
|
|
fn len(&self) -> usize {
|
|
self.info.size()
|
|
}
|
|
|
|
fn resize(&self, _new_size: usize) -> Result<()> {
|
|
Err(Error::new(Errno::EPERM))
|
|
}
|
|
|
|
fn metadata(&self) -> Metadata {
|
|
self.info.metadata()
|
|
}
|
|
|
|
fn ino(&self) -> u64 {
|
|
self.info.ino()
|
|
}
|
|
|
|
fn type_(&self) -> InodeType {
|
|
InodeType::File
|
|
}
|
|
|
|
fn mode(&self) -> InodeMode {
|
|
self.info.mode()
|
|
}
|
|
|
|
fn set_mode(&self, mode: InodeMode) {
|
|
self.info.set_mode(mode)
|
|
}
|
|
|
|
fn atime(&self) -> Duration {
|
|
self.info.atime()
|
|
}
|
|
|
|
fn set_atime(&self, time: Duration) {
|
|
self.info.set_atime(time)
|
|
}
|
|
|
|
fn mtime(&self) -> Duration {
|
|
self.info.mtime()
|
|
}
|
|
|
|
fn set_mtime(&self, time: Duration) {
|
|
self.info.set_mtime(time)
|
|
}
|
|
|
|
fn read_at(&self, offset: usize, buf: &mut [u8]) -> Result<usize> {
|
|
let data = self.inner.data()?;
|
|
let start = data.len().min(offset);
|
|
let end = data.len().min(offset + buf.len());
|
|
let len = end - start;
|
|
buf[0..len].copy_from_slice(&data[start..end]);
|
|
Ok(len)
|
|
}
|
|
|
|
fn read_direct_at(&self, offset: usize, buf: &mut [u8]) -> Result<usize> {
|
|
self.read_at(offset, buf)
|
|
}
|
|
|
|
fn write_at(&self, _offset: usize, _buf: &[u8]) -> Result<usize> {
|
|
Err(Error::new(Errno::EPERM))
|
|
}
|
|
|
|
fn write_direct_at(&self, _offset: usize, _buf: &[u8]) -> Result<usize> {
|
|
Err(Error::new(Errno::EPERM))
|
|
}
|
|
|
|
fn read_link(&self) -> Result<String> {
|
|
Err(Error::new(Errno::EINVAL))
|
|
}
|
|
|
|
fn write_link(&self, _target: &str) -> Result<()> {
|
|
Err(Error::new(Errno::EINVAL))
|
|
}
|
|
|
|
fn ioctl(&self, _cmd: IoctlCmd, _arg: usize) -> Result<i32> {
|
|
Err(Error::new(Errno::EPERM))
|
|
}
|
|
|
|
fn sync(&self) -> Result<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn fs(&self) -> Arc<dyn FileSystem> {
|
|
self.info.fs().upgrade().unwrap()
|
|
}
|
|
|
|
fn is_dentry_cacheable(&self) -> bool {
|
|
!self.info.is_volatile()
|
|
}
|
|
}
|
|
|
|
pub trait FileOps: Sync + Send {
|
|
fn data(&self) -> Result<Vec<u8>>;
|
|
}
|