123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893 |
- use core::{
- any::Any,
- fmt::Debug,
- sync::atomic::{AtomicBool, Ordering},
- };
- use alloc::{
- collections::LinkedList,
- sync::{Arc, Weak},
- vec::Vec,
- };
- use intertrait::CastFromSync;
- use system_error::SystemError;
- use crate::{
- filesystem::vfs::{
- file::{File, FileMode},
- FilePrivateData, IndexNode, Metadata,
- },
- libs::{
- rbtree::RBTree,
- rwlock::RwLock,
- spinlock::{SpinLock, SpinLockGuard},
- wait_queue::WaitQueue,
- },
- process::ProcessManager,
- sched::{schedule, SchedMode},
- time::{
- timer::{next_n_us_timer_jiffies, Timer, WakeUpHelper},
- PosixTimeSpec,
- },
- };
- pub mod syscall;
- #[derive(Debug, Clone)]
- pub struct LockedEventPoll(Arc<SpinLock<EventPoll>>);
- /// 内核的Epoll对象结构体,当用户创建一个Epoll时,内核就会创建一个该类型对象
- /// 它对应一个epfd
- #[derive(Debug)]
- pub struct EventPoll {
- /// epoll_wait用到的等待队列
- epoll_wq: WaitQueue,
- /// 维护所有添加进来的socket的红黑树
- ep_items: RBTree<i32, Arc<EPollItem>>,
- /// 接收就绪的描述符列表
- ready_list: LinkedList<Arc<EPollItem>>,
- /// 是否已经关闭
- shutdown: AtomicBool,
- self_ref: Option<Weak<SpinLock<EventPoll>>>,
- }
- impl EventPoll {
- pub const EP_MAX_EVENTS: u32 = u32::MAX / (core::mem::size_of::<EPollEvent>() as u32);
- /// 用于获取inode中的epitem队列
- pub const ADD_EPOLLITEM: u32 = 0x7965;
- pub fn new() -> Self {
- Self {
- epoll_wq: WaitQueue::default(),
- ep_items: RBTree::new(),
- ready_list: LinkedList::new(),
- shutdown: AtomicBool::new(false),
- self_ref: None,
- }
- }
- }
- impl Default for EventPoll {
- fn default() -> Self {
- Self::new()
- }
- }
- /// EpollItem表示的是Epoll所真正管理的对象
- /// 每当用户向Epoll添加描述符时都会注册一个新的EpollItem,EpollItem携带了一些被监听的描述符的必要信息
- #[derive(Debug)]
- pub struct EPollItem {
- /// 对应的Epoll
- epoll: Weak<SpinLock<EventPoll>>,
- /// 用户注册的事件
- event: RwLock<EPollEvent>,
- /// 监听的描述符
- fd: i32,
- /// 对应的文件
- file: Weak<File>,
- }
- impl EPollItem {
- pub fn new(
- epoll: Weak<SpinLock<EventPoll>>,
- events: EPollEvent,
- fd: i32,
- file: Weak<File>,
- ) -> Self {
- Self {
- epoll,
- event: RwLock::new(events),
- fd,
- file,
- }
- }
- pub fn epoll(&self) -> Weak<SpinLock<EventPoll>> {
- self.epoll.clone()
- }
- pub fn event(&self) -> &RwLock<EPollEvent> {
- &self.event
- }
- pub fn file(&self) -> Weak<File> {
- self.file.clone()
- }
- pub fn fd(&self) -> i32 {
- self.fd
- }
- /// ## 通过epoll_item来执行绑定文件的poll方法,并获取到感兴趣的事件
- fn ep_item_poll(&self) -> EPollEventType {
- let file = self.file.upgrade();
- if file.is_none() {
- return EPollEventType::empty();
- }
- if let Ok(events) = file.unwrap().poll() {
- let events = events as u32 & self.event.read().events;
- return EPollEventType::from_bits_truncate(events);
- }
- return EPollEventType::empty();
- }
- }
- pub trait KernelIoctlData: Send + Sync + Any + Debug + CastFromSync {}
- impl KernelIoctlData for EPollItem {}
- /// ### Epoll文件的私有信息
- #[derive(Debug, Clone)]
- pub struct EPollPrivateData {
- epoll: LockedEventPoll,
- }
- /// ### 该结构体将Epoll加入文件系统
- #[derive(Debug)]
- pub struct EPollInode {
- epoll: LockedEventPoll,
- }
- impl EPollInode {
- pub fn new(epoll: LockedEventPoll) -> Arc<Self> {
- Arc::new(Self { epoll })
- }
- }
- impl IndexNode for EPollInode {
- fn read_at(
- &self,
- _offset: usize,
- _len: usize,
- _buf: &mut [u8],
- _data: SpinLockGuard<FilePrivateData>,
- ) -> Result<usize, SystemError> {
- Err(SystemError::ENOSYS)
- }
- fn write_at(
- &self,
- _offset: usize,
- _len: usize,
- _buf: &[u8],
- _data: SpinLockGuard<FilePrivateData>,
- ) -> Result<usize, SystemError> {
- Err(SystemError::ENOSYS)
- }
- fn poll(&self, _private_data: &FilePrivateData) -> Result<usize, SystemError> {
- // 需要实现epoll嵌套epoll时,需要实现这里
- todo!()
- }
- fn fs(&self) -> Arc<dyn crate::filesystem::vfs::FileSystem> {
- todo!()
- }
- fn as_any_ref(&self) -> &dyn core::any::Any {
- self
- }
- fn list(&self) -> Result<Vec<alloc::string::String>, SystemError> {
- Err(SystemError::ENOSYS)
- }
- fn metadata(&self) -> Result<Metadata, SystemError> {
- Ok(Metadata::default())
- }
- fn close(&self, _data: SpinLockGuard<FilePrivateData>) -> Result<(), SystemError> {
- // 释放资源
- let mut epoll = self.epoll.0.lock_irqsave();
- // 唤醒epoll上面等待的所有进程
- epoll.shutdown.store(true, Ordering::SeqCst);
- epoll.ep_wake_all();
- let fds = epoll.ep_items.keys().cloned().collect::<Vec<_>>();
- // 清理红黑树里面的epitems
- for fd in fds {
- let file = ProcessManager::current_pcb()
- .fd_table()
- .read()
- .get_file_by_fd(fd);
- if file.is_some() {
- file.unwrap().remove_epoll(&Arc::downgrade(&self.epoll.0))?;
- }
- epoll.ep_items.remove(&fd);
- }
- Ok(())
- }
- fn open(
- &self,
- _data: SpinLockGuard<FilePrivateData>,
- _mode: &FileMode,
- ) -> Result<(), SystemError> {
- Ok(())
- }
- }
- impl EventPoll {
- /// ## 创建epoll对象
- ///
- /// ### 参数
- /// - flags: 创建的epoll文件的FileMode
- ///
- /// ### 返回值
- /// - 成功则返回Ok(fd),否则返回Err
- pub fn do_create_epoll(flags: FileMode) -> Result<usize, SystemError> {
- if !flags.difference(FileMode::O_CLOEXEC).is_empty() {
- return Err(SystemError::EINVAL);
- }
- // 创建epoll
- let epoll = LockedEventPoll(Arc::new(SpinLock::new(EventPoll::new())));
- epoll.0.lock_irqsave().self_ref = Some(Arc::downgrade(&epoll.0));
- // 创建epoll的inode对象
- let epoll_inode = EPollInode::new(epoll.clone());
- let mut ep_file = File::new(
- epoll_inode,
- FileMode::O_RDWR | (flags & FileMode::O_CLOEXEC),
- )?;
- // 设置ep_file的FilePrivateData
- ep_file.private_data = SpinLock::new(FilePrivateData::EPoll(EPollPrivateData { epoll }));
- let current_pcb = ProcessManager::current_pcb();
- let fd_table = current_pcb.fd_table();
- let mut fd_table_guard = fd_table.write();
- let fd = fd_table_guard.alloc_fd(ep_file, None)?;
- Ok(fd as usize)
- }
- /// ## epoll_ctl的具体实现
- ///
- /// 根据不同的op对epoll文件进行增删改
- ///
- /// ### 参数
- /// - epfd: 操作的epoll文件描述符
- /// - op: 对应的操作
- /// - fd: 操作对应的文件描述符
- /// - epds: 从用户态传入的event,若op为EpollCtlAdd,则对应注册的监听事件,若op为EPollCtlMod,则对应更新的事件,删除操作不涉及此字段
- /// - nonblock: 定义这次操作是否为非阻塞(有可能其他地方占有EPoll的锁)
- pub fn do_epoll_ctl(
- epfd: i32,
- op: EPollCtlOption,
- fd: i32,
- epds: &mut EPollEvent,
- nonblock: bool,
- ) -> Result<usize, SystemError> {
- let current_pcb = ProcessManager::current_pcb();
- let fd_table = current_pcb.fd_table();
- let fd_table_guard = fd_table.read();
- // 获取epoll和对应fd指向的文件
- let ep_file = fd_table_guard
- .get_file_by_fd(epfd)
- .ok_or(SystemError::EBADF)?;
- let dst_file = fd_table_guard
- .get_file_by_fd(fd)
- .ok_or(SystemError::EBADF)?;
- // 检查是否允许 EPOLLWAKEUP
- if op != EPollCtlOption::Del {
- epds.events &= !EPollEventType::EPOLLWAKEUP.bits();
- }
- let events = EPollEventType::from_bits_truncate(epds.events);
- // 检查获取到的两个文件的正确性
- // 首先是不能自己嵌套自己
- // 然后ep_file必须是epoll文件
- if Arc::ptr_eq(&ep_file, &dst_file) || !Self::is_epoll_file(&ep_file) {
- return Err(SystemError::EINVAL);
- }
- if op != EPollCtlOption::Del && events.contains(EPollEventType::EPOLLEXCLUSIVE) {
- // epoll独占模式下不允许EpollCtlMod
- if op == EPollCtlOption::Mod {
- return Err(SystemError::EINVAL);
- }
- // 不支持嵌套的独占唤醒
- if op == EPollCtlOption::Add && Self::is_epoll_file(&dst_file)
- || !events
- .difference(EPollEventType::EPOLLEXCLUSIVE_OK_BITS)
- .is_empty()
- {
- return Err(SystemError::EINVAL);
- }
- }
- // 从FilePrivateData获取到epoll
- if let FilePrivateData::EPoll(epoll_data) = &*ep_file.private_data.lock() {
- let mut epoll_guard = {
- if nonblock {
- // 如果设置非阻塞,则尝试获取一次锁
- if let Ok(guard) = epoll_data.epoll.0.try_lock_irqsave() {
- guard
- } else {
- return Err(SystemError::EAGAIN_OR_EWOULDBLOCK);
- }
- } else {
- epoll_data.epoll.0.lock_irqsave()
- }
- };
- if op == EPollCtlOption::Add {
- // TODO: 循环检查是否为epoll嵌套epoll的情况,如果是则需要检测其深度
- // 这里是需要一种检测算法的,但是目前未考虑epoll嵌套epoll的情况,所以暂时未实现
- // Linux算法:https://code.dragonos.org.cn/xref/linux-6.1.9/fs/eventpoll.c?r=&mo=56953&fi=2057#2133
- if Self::is_epoll_file(&dst_file) {
- todo!();
- }
- }
- let ep_item = epoll_guard.ep_items.get(&fd);
- match op {
- EPollCtlOption::Add => {
- // 如果已经存在,则返回错误
- if ep_item.is_some() {
- return Err(SystemError::EEXIST);
- }
- // 设置epoll
- let epitem = Arc::new(EPollItem::new(
- Arc::downgrade(&epoll_data.epoll.0),
- *epds,
- fd,
- Arc::downgrade(&dst_file),
- ));
- Self::ep_insert(&mut epoll_guard, dst_file, epitem)?;
- }
- EPollCtlOption::Del => {
- // 不存在则返回错误
- if ep_item.is_none() {
- return Err(SystemError::ENOENT);
- }
- // 删除
- Self::ep_remove(&mut epoll_guard, fd, Some(dst_file))?;
- }
- EPollCtlOption::Mod => {
- // 不存在则返回错误
- if ep_item.is_none() {
- return Err(SystemError::ENOENT);
- }
- let ep_item = ep_item.unwrap().clone();
- if ep_item.event.read().events & EPollEventType::EPOLLEXCLUSIVE.bits() != 0 {
- epds.events |=
- EPollEventType::EPOLLERR.bits() | EPollEventType::EPOLLHUP.bits();
- Self::ep_modify(&mut epoll_guard, ep_item, epds)?;
- }
- }
- }
- }
- Ok(0)
- }
- /// ## epoll_wait的具体实现
- pub fn do_epoll_wait(
- epfd: i32,
- epoll_event: &mut [EPollEvent],
- max_events: i32,
- timespec: Option<PosixTimeSpec>,
- ) -> Result<usize, SystemError> {
- let current_pcb = ProcessManager::current_pcb();
- let fd_table = current_pcb.fd_table();
- let fd_table_guard = fd_table.read();
- // 获取epoll文件
- let ep_file = fd_table_guard
- .get_file_by_fd(epfd)
- .ok_or(SystemError::EBADF)?;
- drop(fd_table_guard);
- // 确保是epoll file
- if !Self::is_epoll_file(&ep_file) {
- return Err(SystemError::EINVAL);
- }
- // 从epoll文件获取到epoll
- let mut epolldata = None;
- if let FilePrivateData::EPoll(epoll_data) = &*ep_file.private_data.lock() {
- epolldata = Some(epoll_data.clone())
- }
- if let Some(epoll_data) = epolldata {
- let epoll = epoll_data.epoll.clone();
- let epoll_guard = epoll.0.lock_irqsave();
- let mut timeout = false;
- if let Some(timespec) = timespec {
- if !(timespec.tv_sec > 0 || timespec.tv_nsec > 0) {
- // 非阻塞情况
- timeout = true;
- }
- }
- // 判断epoll上有没有就绪事件
- let mut available = epoll_guard.ep_events_available();
- drop(epoll_guard);
- loop {
- if available {
- // 如果有就绪的事件,则直接返回就绪事件
- return Self::ep_send_events(epoll.clone(), epoll_event, max_events);
- }
- if epoll.0.lock_irqsave().shutdown.load(Ordering::SeqCst) {
- // 如果已经关闭
- return Err(SystemError::EBADF);
- }
- // 如果超时
- if timeout {
- return Ok(0);
- }
- // 自旋等待一段时间
- available = {
- let mut ret = false;
- for _ in 0..50 {
- if let Ok(guard) = epoll.0.try_lock_irqsave() {
- if guard.ep_events_available() {
- ret = true;
- break;
- }
- }
- }
- // 最后再次不使用try_lock尝试
- if !ret {
- ret = epoll.0.lock_irqsave().ep_events_available();
- }
- ret
- };
- if available {
- continue;
- }
- // 如果有未处理的信号则返回错误
- if current_pcb.has_pending_signal_fast() {
- return Err(SystemError::ERESTARTSYS);
- }
- // 还未等待到事件发生,则睡眠
- // 注册定时器
- let mut timer = None;
- if let Some(timespec) = timespec {
- let handle = WakeUpHelper::new(current_pcb.clone());
- let jiffies = next_n_us_timer_jiffies(
- (timespec.tv_sec * 1000000 + timespec.tv_nsec / 1000) as u64,
- );
- let inner: Arc<Timer> = Timer::new(handle, jiffies);
- inner.activate();
- timer = Some(inner);
- }
- let guard = epoll.0.lock_irqsave();
- // 睡眠,等待事件发生
- // 如果wq已经dead,则直接返回错误
- unsafe { guard.epoll_wq.sleep_without_schedule() }.inspect_err(|_| {
- if let Some(timer) = timer.as_ref() {
- timer.cancel();
- }
- })?;
- drop(guard);
- schedule(SchedMode::SM_NONE);
- // 被唤醒后,检查是否有事件可读
- available = epoll.0.lock_irqsave().ep_events_available();
- if let Some(timer) = timer {
- if timer.as_ref().timeout() {
- // 超时
- timeout = true;
- } else {
- // 未超时,则取消计时器
- timer.cancel();
- }
- }
- }
- } else {
- panic!("An epoll file does not have the corresponding private information");
- }
- }
- /// ## 将已经准备好的事件拷贝到用户空间
- ///
- /// ### 参数
- /// - epoll: 对应的epoll
- /// - user_event: 用户空间传入的epoll_event地址,因为内存对其问题,所以这里需要直接操作地址
- /// - max_events: 处理的最大事件数量
- fn ep_send_events(
- epoll: LockedEventPoll,
- user_event: &mut [EPollEvent],
- max_events: i32,
- ) -> Result<usize, SystemError> {
- let mut ep_guard = epoll.0.lock_irqsave();
- let mut res: usize = 0;
- // 在水平触发模式下,需要将epitem再次加入队列,在下次循环再次判断是否还有事件
- // (所以边缘触发的效率会高于水平触发,但是水平触发某些情况下能够使得更迅速地向用户反馈)
- let mut push_back = Vec::new();
- while let Some(epitem) = ep_guard.ready_list.pop_front() {
- if res >= max_events as usize {
- push_back.push(epitem);
- break;
- }
- let ep_events = EPollEventType::from_bits_truncate(epitem.event.read().events);
- // 再次poll获取事件(为了防止水平触发一直加入队列)
- let revents = epitem.ep_item_poll();
- if revents.is_empty() {
- continue;
- }
- // 构建触发事件结构体
- let event = EPollEvent {
- events: revents.bits,
- data: epitem.event.read().data,
- };
- // 这里是需要判断下一个写入的位置是否为空指针
- // TODO:这里有可能会出现事件丢失的情况
- // 如果用户传入的数组长度小于传入的max_event,到这里时如果已经到数组最大长度,但是未到max_event
- // 会出现的问题是我们会把这个数据写入到后面的内存中,用户无法在传入的数组中拿到事件,而且写脏数据到了后面一片内存,导致事件丢失
- // 出现这个问题的几率比较小,首先是因为用户的使用不规范,后因为前面地址校验是按照max_event来校验的,只会在两块内存连着分配时出现,但是也是需要考虑的
- // 以下的写法判断并无意义,只是记一下错误处理
- // offset += core::mem::size_of::<EPollEvent>();
- // if offset >= max_offset {
- // // 当前指向的地址已为空,则把epitem放回队列
- // ep_guard.ready_list.push_back(epitem.clone());
- // if res == 0 {
- // // 一个都未写入成功,表明用户传进的地址就是有问题的
- // return Err(SystemError::EFAULT);
- // }
- // }
- // 拷贝到用户空间
- user_event[res] = event;
- // 记数加一
- res += 1;
- // crate::debug!("ep send {event:?}");
- if ep_events.contains(EPollEventType::EPOLLONESHOT) {
- let mut event_writer = epitem.event.write();
- let new_event = event_writer.events & EPollEventType::EP_PRIVATE_BITS.bits;
- event_writer.set_events(new_event);
- } else if !ep_events.contains(EPollEventType::EPOLLET) {
- push_back.push(epitem);
- }
- }
- for item in push_back {
- ep_guard.ep_add_ready(item);
- }
- Ok(res)
- }
- // ### 查看文件是否为epoll文件
- fn is_epoll_file(file: &Arc<File>) -> bool {
- if let FilePrivateData::EPoll(_) = *file.private_data.lock() {
- return true;
- }
- return false;
- }
- fn ep_insert(
- epoll_guard: &mut SpinLockGuard<EventPoll>,
- dst_file: Arc<File>,
- epitem: Arc<EPollItem>,
- ) -> Result<(), SystemError> {
- if Self::is_epoll_file(&dst_file) {
- return Err(SystemError::ENOSYS);
- // TODO:现在的实现先不考虑嵌套其它类型的文件(暂时只针对socket),这里的嵌套指epoll/select/poll
- }
- let test_poll = dst_file.poll();
- if test_poll.is_err() && test_poll.unwrap_err() == SystemError::EOPNOTSUPP_OR_ENOTSUP {
- // 如果目标文件不支持poll
- return Err(SystemError::ENOSYS);
- }
- epoll_guard.ep_items.insert(epitem.fd, epitem.clone());
- // 检查文件是否已经有事件发生
- let event = epitem.ep_item_poll();
- if !event.is_empty() {
- // 加入到就绪队列
- epoll_guard.ep_add_ready(epitem.clone());
- epoll_guard.ep_wake_one();
- }
- // TODO: 嵌套epoll?
- // 这个标志是用与电源管理相关,暂时不支持
- if epitem.event.read().events & EPollEventType::EPOLLWAKEUP.bits() != 0 {
- return Err(SystemError::ENOSYS);
- }
- dst_file.add_epoll(epitem.clone())?;
- Ok(())
- }
- pub fn ep_remove(
- epoll: &mut SpinLockGuard<EventPoll>,
- fd: i32,
- dst_file: Option<Arc<File>>,
- ) -> Result<(), SystemError> {
- if let Some(dst_file) = dst_file {
- dst_file.remove_epoll(epoll.self_ref.as_ref().unwrap())?;
- }
- let epitem = epoll.ep_items.remove(&fd).unwrap();
- let _ = epoll
- .ready_list
- .extract_if(|item| Arc::ptr_eq(item, &epitem));
- Ok(())
- }
- /// ## 修改已经注册的监听事件
- ///
- /// ### 参数
- /// - epoll_guard: EventPoll的锁
- /// - epitem: 需要修改的描述符对应的epitem
- /// - event: 新的事件
- fn ep_modify(
- epoll_guard: &mut SpinLockGuard<EventPoll>,
- epitem: Arc<EPollItem>,
- event: &EPollEvent,
- ) -> Result<(), SystemError> {
- let mut epi_event_guard = epitem.event.write();
- // 修改epitem
- epi_event_guard.events = event.events;
- epi_event_guard.data = event.data;
- drop(epi_event_guard);
- // 修改后检查文件是否已经有感兴趣事件发生
- let event = epitem.ep_item_poll();
- if !event.is_empty() {
- epoll_guard.ep_add_ready(epitem.clone());
- epoll_guard.ep_wake_one();
- }
- // TODO:处理EPOLLWAKEUP,目前不支持
- Ok(())
- }
- /// ### 判断epoll是否有就绪item
- pub fn ep_events_available(&self) -> bool {
- !self.ready_list.is_empty()
- }
- /// ### 将epitem加入到就绪队列,如果为重复添加则忽略
- pub fn ep_add_ready(&mut self, epitem: Arc<EPollItem>) {
- let ret = self.ready_list.iter().find(|epi| Arc::ptr_eq(epi, &epitem));
- if ret.is_none() {
- self.ready_list.push_back(epitem);
- }
- }
- /// ### 判断该epoll上是否有进程在等待
- pub fn ep_has_waiter(&self) -> bool {
- self.epoll_wq.len() != 0
- }
- /// ### 唤醒所有在epoll上等待的进程
- pub fn ep_wake_all(&self) {
- self.epoll_wq.wakeup_all(None);
- }
- /// ### 唤醒所有在epoll上等待的首个进程
- pub fn ep_wake_one(&self) {
- self.epoll_wq.wakeup(None);
- }
- /// ### epoll的回调,支持epoll的文件有事件到来时直接调用该方法即可
- pub fn wakeup_epoll(
- epitems: &SpinLock<LinkedList<Arc<EPollItem>>>,
- pollflags: Option<EPollEventType>,
- ) -> Result<(), SystemError> {
- let mut epitems_guard = epitems.try_lock_irqsave()?;
- // 一次只取一个,因为一次也只有一个进程能拿到对应文件的🔓
- if let Some(epitem) = epitems_guard.pop_front() {
- let pollflags = pollflags.unwrap_or({
- if let Some(file) = epitem.file.upgrade() {
- EPollEventType::from_bits_truncate(file.poll()? as u32)
- } else {
- EPollEventType::empty()
- }
- });
- if let Some(epoll) = epitem.epoll().upgrade() {
- let mut epoll_guard = epoll.try_lock()?;
- let binding = epitem.clone();
- let event_guard = binding.event().read();
- let ep_events = EPollEventType::from_bits_truncate(event_guard.events());
- // 检查事件合理性以及是否有感兴趣的事件
- if !(ep_events
- .difference(EPollEventType::EP_PRIVATE_BITS)
- .is_empty()
- || pollflags.difference(ep_events).is_empty())
- {
- // TODO: 未处理pm相关
- // 首先将就绪的epitem加入等待队列
- epoll_guard.ep_add_ready(epitem.clone());
- if epoll_guard.ep_has_waiter() {
- if ep_events.contains(EPollEventType::EPOLLEXCLUSIVE)
- && !pollflags.contains(EPollEventType::POLLFREE)
- {
- // 避免惊群
- epoll_guard.ep_wake_one();
- } else {
- epoll_guard.ep_wake_all();
- }
- }
- }
- epitems_guard.push_back(epitem);
- }
- }
- Ok(())
- }
- }
- /// 与C兼容的Epoll事件结构体
- #[derive(Copy, Clone, Default)]
- #[repr(packed)]
- #[repr(C)]
- pub struct EPollEvent {
- /// 表示触发的事件
- events: u32,
- /// 内核态不使用该字段,该字段由用户态自由使用,在事件发生时内核将会原样返回
- data: u64,
- }
- impl Debug for EPollEvent {
- fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
- let events = self.events;
- let u64 = self.data;
- f.debug_struct("epoll_event")
- .field("events", &events)
- .field("data", &u64)
- .finish()
- }
- }
- impl EPollEvent {
- pub fn set_events(&mut self, events: u32) {
- self.events = events;
- }
- pub fn events(&self) -> u32 {
- self.events
- }
- }
- /// ## epoll_ctl函数的参数
- #[derive(Debug, PartialEq)]
- pub enum EPollCtlOption {
- /// 注册新的文件描述符到epfd
- Add,
- /// 将对应的文件描述符从epfd中删除
- Del,
- /// 修改已经注册的文件描述符的监听事件
- Mod,
- }
- impl EPollCtlOption {
- pub fn from_op_num(op: usize) -> Result<Self, SystemError> {
- match op {
- 1 => Ok(Self::Add),
- 2 => Ok(Self::Del),
- 3 => Ok(Self::Mod),
- _ => Err(SystemError::EINVAL),
- }
- }
- }
- bitflags! {
- #[allow(dead_code)]
- pub struct EPollEventType: u32 {
- /// 对应的描述符有新的数据可读时会触发
- const EPOLLIN = 0x00000001;
- /// 对应的描述符有紧急数据可读时会触发
- const EPOLLPRI = 0x00000002;
- /// 对应的描述符可以写入数据时会触发
- const EPOLLOUT = 0x00000004;
- /// 对应的描述符发生错误时会触发
- const EPOLLERR = 0x00000008;
- /// 对应的描述符被挂断(连接关闭)时会触发
- const EPOLLHUP = 0x00000010;
- /// 对应的描述符不是一个有效的文件描述符时会触发
- const EPOLLNVAL = 0x00000020;
- /// 普通数据可读,类似于`EPOLLIN`
- const EPOLLRDNORM = 0x00000040;
- /// 优先级带外数据可读
- const EPOLLRDBAND = 0x00000080;
- /// 普通数据可写,类似于'EPOLLOUT'
- const EPOLLWRNORM = 0x00000100;
- /// 优先级带外数据可写
- const EPOLLWRBAND = 0x00000200;
- /// 通过消息队列收到消息时会触
- const EPOLLMSG = 0x00000400;
- /// 对应的描述符被挂断(连接关闭)的一端发送了 FIN 时会触发(读关闭)
- const EPOLLRDHUP = 0x00002000;
- /// 以下为额外选项
- ///
- /// 特定选项,用于异步 I/O,目前未实现
- const EPOLL_URING_WAKE = 1u32 << 27;
- /// 设置epoll为独占模式
- const EPOLLEXCLUSIVE = 1u32 << 28;
- /// 允许在系统挂起时唤醒 epoll,通常用于通过 eventfd 或 timerfd 唤醒 epoll,(通常与电源管理相关,未实现)
- const EPOLLWAKEUP = 1u32 << 29;
- /// 表示只监听一次事件,之后需要重新添加
- const EPOLLONESHOT = 1u32 << 30;
- /// 启用边缘触发模式(即只有下次触发事件时才会通过epoll_wait返回),
- /// 对应为水平触发(默认),水平触发模式下若这次未处理完数据,那epoll还会将其加入自己的就绪队列
- const EPOLLET = 1u32 << 31;
- /// 以下为组合码
- const EPOLLINOUT_BITS = Self::EPOLLIN.bits | Self::EPOLLOUT.bits;
- const EPOLLEXCLUSIVE_OK_BITS =
- Self::EPOLLINOUT_BITS.bits
- | Self::EPOLLERR.bits
- | Self::EPOLLHUP.bits
- | Self::EPOLLWAKEUP.bits
- | Self::EPOLLET.bits
- | Self::EPOLLEXCLUSIVE.bits;
- const EP_PRIVATE_BITS =
- Self::EPOLLWAKEUP.bits
- | Self::EPOLLONESHOT.bits
- | Self::EPOLLET.bits
- | Self::EPOLLEXCLUSIVE.bits;
- /// 表示epoll已经被释放,但是在目前的设计中未用到
- const POLLFREE = 0x4000;
- /// listen状态的socket可以接受连接
- const EPOLL_LISTEN_CAN_ACCEPT = Self::EPOLLIN.bits | Self::EPOLLRDNORM.bits;
- }
- }
|