use std::cell::RefCell; use std::ops::{Deref, DerefMut}; use std::rc::Weak; use syscall::error::{Error, Result, EBADF, EINVAL, EPIPE}; use syscall::flag::{EventFlags, F_GETFL, F_SETFL, O_ACCMODE}; use crate::pty::Pty; use crate::resource::Resource; /// Read side of a pipe pub struct PtyTermios { pty: Weak>, flags: usize, } impl PtyTermios { pub fn new(pty: Weak>, flags: usize) -> Self { PtyTermios { pty, flags } } } impl Resource for PtyTermios { fn pty(&self) -> Weak> { self.pty.clone() } fn flags(&self) -> usize { self.flags } fn path(&mut self, buf: &mut [u8]) -> Result { if let Some(pty_lock) = self.pty.upgrade() { pty_lock.borrow_mut().path(buf) } else { Err(Error::new(EPIPE)) } } fn read(&mut self, buf: &mut [u8]) -> Result { if let Some(pty_lock) = self.pty.upgrade() { let pty = pty_lock.borrow(); let termios: &[u8] = pty.termios.deref(); let mut i = 0; while i < buf.len() && i < termios.len() { buf[i] = termios[i]; i += 1; } Ok(i) } else { Ok(0) } } fn write(&mut self, buf: &[u8]) -> Result { if let Some(pty_lock) = self.pty.upgrade() { let mut pty = pty_lock.borrow_mut(); let termios: &mut [u8] = pty.termios.deref_mut(); let mut i = 0; while i < buf.len() && i < termios.len() { termios[i] = buf[i]; i += 1; } Ok(i) } else { Err(Error::new(EPIPE)) } } fn sync(&mut self) -> Result<()> { Ok(()) } fn fcntl(&mut self, cmd: usize, arg: usize) -> Result { match cmd { F_GETFL => Ok(self.flags), F_SETFL => { self.flags = (self.flags & O_ACCMODE) | (arg & !O_ACCMODE); Ok(0) } _ => Err(Error::new(EINVAL)), } } fn fevent(&mut self) -> Result { Err(Error::new(EBADF)) } fn events(&mut self) -> EventFlags { EventFlags::empty() } }