epoll源码分析(二)

上面的原型是epollfd所维护的主结构,下面是每一个具体的fd结构.

以后每一个fd加入到epoll中,就会创建一个struct epitem结构,并插入至红黑树中。

接着是epoll_ctl函数原型:

  1. asmlinkage long sys_epoll_ctl(int epfd,int op,int fd,struct epoll_event __user *event)
  2. {
  3.     int error;
  4.     struct file *file,*tfile;
  5.     struct eventpoll *ep;
  6.     struct epoll_event epds;
  7.     
  8.     error = -FAULT;
  9.     //判断行参的合法性
  10.     if(ep_op_has_event(op) && copy_from_user(&epds,event,sizeof(struct         epoll_event)))
  11.             goto error_return;
  12.  
  13.     error = -EBADF;
  14.     file = fget (epfd);
  15.     if(!file)    goto error_return;
  16.     
  17.     tfile = fget(fd);
  18.     if(!tfile)    goto error_fput;
  19.     
  20.     error = -EPERM;
  21.     //不能没有poll驱动
  22.     if(!tfile->f_op || !tfile->f_op->poll)
  23.         goto error_tgt_fput;
  24.         
  25.     error =-EINVAL;
  26.     //防止自己监听自己
  27.     if(file == tfile || !is_file_poll(file))
  28.         goto error_tgt_fput;
  29.     //在create时存入进去的,现在将其拿出来
  30.     ep = file->private->data;
  31.     
  32.     mutex_lock(&ep->mtx);
  33.     //防止重复添加
  34.     epi = epi_find(ep,tfile,fd);
  35.     error = -EINVAL;
  36.     
  37.     switch(op)
  38.     {
  39.         ….....
  40.         case EPOLL_CTL_ADD:
  41.             if(!epi)
  42.             {
  43.                 epds.events |=EPOLLERR | POLLHUP;
  44.                 error = ep_insert(ep,&epds,tfile,fd);
  45.             } else 
  46.                 error = -EEXIST;
  47.             break;
  48.         …....
  49.     }
  50.     return error;    
  51. }

下面就是插入代码:

  1. static int ep_insert(struct eventpoll *ep,struct epoll_event *event,
  2.                 struct file *tfile,int fd)
  3. {
  4.     int error ,revents,pwake = 0;
  5.     unsigned long flags ;
  6.     struct epitem *epi;
  7.     /*
  8.     struct ep_queue{
  9.         poll_table pt;
  10.         struct epitem *epi;
  11.     }
  12.     */
  13.     struct ep_pqueue epq;
  14.     
  15.     //分配一个epitem结构体来保存每个加入的fd
  16.     error = -ENOMEM;
  17.     if(!(epi = kmem_cache_alloc(epi_cache,GFP_KERNEL)))
  18.         goto error_return;
  19.     //初始化该结构体
  20.     ep_rb_initnode(&epi->rbn);
  21.     INIT_LIST_HEAD(&epi->rdllink);
  22.     INIT_LIST_HEAD(&epi->fllink);
  23.     INIT_LIST_HEAD(&epi->pwqlist);
  24.     epi->ep = ep;
  25.     ep_set_ffd(&epi->ffd,tfile,fd);
  26.     epi->event = *event;
  27.     epi->nwait = 0;
  28.     epi->next = EP_UNACTIVE_PTR;
  29.     
  30.     epq.epi = epi;
  31.     //安装poll回调函数
  32.     init_poll_funcptr(&epq.pt,ep_ptable_queue_proc);
  33.     //调用poll函数来获取当前事件位,其实是利用它来调用注册函数ep_ptable_queue_proc
  34.     revents = tfile->f_op->poll(tfile,&epq.pt);
  35.  
  36.     if(epi->nwait < 0)
  37.         goto error_unregister;
  38.  
  39.     spin_lock(&tfile->f_ep_lock);
  40.     list_add_tail(&epi->fllink,&tfile->f_ep_lilnks);
  41.     spin_unlock(&tfile->f_ep_lock);
  42.     
  43.     ep_rbtree_insert(ep,epi);
  44.     spin_lock_irqsave(&ep->lock,flags);
  45.  
  46.     if((revents & event->events) && !ep_is_linked(&epi->rdllink))
  47.     {
  48.         list_add_tail(&epi->rdllink,&ep->rdllist);
  49.     if(waitqueue_active(&ep->wq))
  50.         __wake_up_locked(&ep->wq,TAKS_UNINTERRUPTIBLE | TASK_INTERRUPTIBLE);
  51.  
  52.     if(waitqueue_active(&ep->poll_wait))
  53.         pwake++;
  54.     }
  55.     
  56.     spin_unlock_irqrestore(&ep->lock,flags);
  57.     if(pwake) 
  58.         ep_poll_safewake(&psw,&ep->poll_wait);
  59.         …....
  60.     
  61.     return 0;
  62.     
  63.     …...
  64.     
  65. }


  1. //当poll醒来时就回调用该函数
  2. static void ep_ptable_queue_proc(struct file *file,wait_queue_head_t *whead,
  3.                 poll_table *pt)
  4. {
  5.     //从注册时的结构中struct ep_pqueue中获取项epi
  6.     struct epitem *epi = ep_item_from_epqueue(pt);
  7.     /*//epitem的私有项,通过pwqlist来进行链接
  8.      *struct eppoll_entry
  9.      {
  10.         struct list_head llink;
  11.         void *base;
  12.         wait_queue_t wait;
  13.         wait_queue_head_t *whead;
  14.      }
  15.     */
  16.     struct eppoll_entry *pwq;//struct epitem的私有项,为每一个fd保存内核poll
  17.  
  18.     //为每一个等待的结构分配一项
  19.     if(epi->nwait >= 0 && (pwq = kmem_cache_alloc(pwq_cache,
  20.             GFP_KERNEL)))
  21.     {
  22.         //醒来就调用ep_poll_callback,这里才是真正意义上的poll醒来时的回调函数
  23.         init_waitqueue_func_entry(&pwq->wait,ep_poll_callback);
  24.         pwq->whead = whead;
  25.         pwq->base = epi;
  26.         //加入到该驱动的等待队列
  27.         add_wait_queue(whead,&pwq->wait);
  28.         //将等待链接也放入到epitem链表中去
  29.         list_add_tail(&pwq->llink,&epi->pwqlist);
  30.         epi->nwait ++;        
  31.     } else {
  32.         epi->nwait = -1;
  33.     }
  34. }
  35. //当poll监听的事件到达时,就会调用下面的函数
  36. static int ep_poll_callback(wait_queue_t *wait,unsigned mode,int sync,void *key)
  37. {
  38.     int pwake = 0;
  39.     unsigned long flags;
  40.     struct epitem *epi = ep_item_from_wait(wait);
  41.     struct eventpoll *ep = epi->ep;
  42.     
  43.     spin_lock_irqsave(&ep->lock,flags);
  44.     //判断注册的感兴趣事件 
  45.     //#define EP_PRIVATE_BITS      (EPOLLONESHOT | EPOLLET)
  46.     //有非EPOLLONESHONT或EPOLLET事件
  47.     if(!(epi->event.events & ~EP_PRIVATE_BITS))
  48.             goto out_unlock;
  49.     
  50.     if(unlikely(ep->ovflist != EP_UNACTIVE_PTR))
  51.     {
  52.         if(epi->next == EP_UNACTIVE_PTR) {
  53.             epi->next = ep->ovflist;
  54.             ep->ovflist = epi;
  55.         }
  56.         goto out_unlock;
  57.     }
  58.  
  59.     if(ep_is_linked(&epi->rdllink))
  60.         goto is_linked;
  61.     //关键是这一句,将该fd加入到epoll监听的就绪链表中
  62.     list_add_tail(&epi->rdllink,&ep->rdllist);
  63. is_linked:
  64.     if(waitqueue_active(&ep->wq))
  65.         __wake_up_locked(&ep->wq,TASK_UNINTERRUPTIBLE 
  66.             | TASK_INTERRUPTIBLE);    
  67.     if(waitqueue_active(&ep->poll_wait))
  68.         pwake++;
  69. out_unlock:
  70.     spin_unlock_irqrestore(&ep->lock,flags);
  71.     
  72.     if(pwake)
  73.         ep_poll_safewake(&psw,&ep->poll_wait);
  74.     return 1;
  75. }

这里采用了两级回调方式,流程如下:

目前为止,整个数据结构就可以描述如下:



下载方式:https://pan.quark.cn/s/a4b39357ea24 在纺织制造领域中,纱线的品质水平对最终制成品的整体质量具有决定性作用。 鉴于消费者对于产品规格和样式要求的不断变化,纺织制造工艺的执行过程日益呈现为一种更为复杂的操作体系,进而导致对纱线质量进行预测的任务变得更加困难。 在众多预测技术中,传统的预测手段在面对多变量间相互交织的复杂关系时,往往显得力不从心。 因此,智能计算技术在预测纱线质量的应用场景中逐渐占据核心地位,其中人工神经网络凭借其卓越的非线性映射特性以及自适应学习机制,成为了众多预测方法中的一种重要选择。 在智能计算技术的范畴内,粒子群优化算法(PSO)和反向传播神经网络(BP神经网络)是两种被广泛采用的技术方案。 粒子群优化算法是一种基于群体智能理念的优化技术,它通过模拟鸟类的群体觅食行为来寻求最优解,该算法因其操作简便、执行高效以及具备优秀的全局搜索性能,在函数优化、神经网络训练等多个领域得到了普遍应用。 反向传播神经网络则是一种由多层节点构成的前馈神经网络,它通过误差反向传播的机制来实现网络权重和阈值的动态调整,从而达成学习与预测的目标。 在实际操作层面,反向传播神经网络因其架构设计简洁、实现过程便捷,因此被广泛部署于各类预测和分类任务之中。 然而,该方法也存在一些固有的局限性,例如容易陷入局部最优状态、网络收敛过程缓慢等问题。 而粒子群优化算法在参与神经网络优化时,能够显著增强神经网络的全局搜索性能并提升收敛速度,有效规避神经网络陷入局部最优的困境。 将粒子群优化算法与反向传播神经网络相结合形成的PSO-BP神经网络,通过运用粒子群优化算法对反向传播神经网络的权值和阈值进行精细化调整,能够在预测纱线断裂强度方面,显著提升预测结果的...
植物实例分割数据集 一、基础信息 数据集名称:植物实例分割数据集 图片数量: - 训练集:9,600张图片 - 验证集:913张图片 - 测试集:455张图片 总计:10,968张图片 分类类别:59个类别,对应数字标签0至58,涵盖多种植物状态或特征。 标注格式:YOLO格式,适用于实例分割任务,包含多边形标注点。 数据格式:图像文件,来源于植物图像数据库,适用于计算机视觉任务。 、适用场景 • 农业植物监测AI系统开发:数据集支持实例分割任务,帮助构建能够自动识别植物特定区域并分类的AI模型,辅助农业专家进行精准监测和分析。 • 智能农业应用研发:集成至农业管理平台,提供实时植物状态识别功能,为作物健康管理和优化种植提供数据支持。 • 学术研究与农业创新:支持植物科学与人工智能交叉领域的研究,助力发表高水平农业AI论文。 • 农业教育与培训:数据集可用于农业院校或培训机构,作为学生学习植物图像分析和实例分割技术的重要资源。 三、数据集优势 • 精准标注与多样性:标注采用YOLO格式,确保分割区域定位精确;包含59个类别,覆盖多种植物状态,具有高度多样性。 • 数据量丰富:拥有超过10,000张图像,大规模数据支持模型充分学习和泛化。 • 任务适配性强:标注兼容主流深度学习框架(如YOLO、Mask R-CNN等),可直接用于实例分割任务,并可能扩展到目标检测或分类等任务。
室内物体实例分割数据集 一、基础信息 • 数据集名称:室内物体实例分割数据集 • 图片数量: 训练集:4923张图片 验证集:3926张图片 测试集:985张图片 总计:9834张图片 • 训练集:4923张图片 • 验证集:3926张图片 • 测试集:985张图片 • 总计:9834张图片 • 分类类别: 床 椅子 沙发 灭火器 人 盆栽植物 冰箱 桌子 垃圾桶 电视 • 床 • 椅子 • 沙发 • 灭火器 • 人 • 盆栽植物 • 冰箱 • 桌子 • 垃圾桶 • 电视 • 标注格式:YOLO格式,包含实例分割的多边形标注,适用于实例分割任务。 • 数据格式:图片为常见格式如JPEG或PNG。 、适用场景 • 实例分割模型开发:适用于训练和评估实例分割AI模型,用于精确识别和分割室内环境中的物体,如家具、电器和人物。 • 智能家居与物联网:可集成到智能家居系统中,实现自动物体检测和场景理解,提升家居自动化水平。 • 机器人导航与交互:支持机器人在室内环境中的物体识别、避障和交互任务,增强机器人智能化应用。 • 学术研究与教育:用于计算机视觉领域实例分割算法的研究与教学,助力AI模型创新与验证。 三、数据集优势 • 类别多样性:涵盖10个常见室内物体类别,包括家具、电器、人物和日常物品,提升模型在多样化场景中的泛化能力。 • 精确标注质量:采用YOLO格式的多边形标注,确保实例分割边界的准确性,适用于精细的物体识别任务。 • 数据规模充足:提供近万张标注图片,满足模型训练、验证和测试的需求,支持稳健的AI开发。 • 任务适配性强:标注格式兼容主流深度学习框架(如YOLO系列),便于快速集成到实例分割项目中,提高开发效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值