热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Linux时钟处理机制(二)

文章标题:Linux时钟处理机制(二)。Linux是中国IT实验室的一个技术频道。包含桌面应用,Linux系统管理,内核研究,嵌入式系统和开源等一些基本分类

  3.3 添加或删除软件时钟

  在了解了软件时钟的数据组织关系之后,现在来看一下如何添加以及删除一个软件时钟。

  3.3.1 添加软件时钟

  在 Linux 内核中要添加一个软件时钟,首先必须分配 struct timer_list 类型的变量,然后调用函数 add_timer() 将该软件时钟添加到相应调用 add_timer 函数的 CPU 的 base 中。 Add_timer 是对函数 __mod_timer() 的一层包装。函数 __mod_timer() 的代码如清单3-2:

  清单3-2 __mod_timer 函数

  int __mod_timer(struct timer_list *timer, unsigned long expires){    struct tvec_base *base, *new_base;    unsigned long flags;    int ret = 0;    ……    base = lock_timer_base(timer, &flags);    if (timer_pending(timer)) {        detach_timer(timer, 0);        ret = 1;    }    new_base = __get_cpu_var(tvec_bases);    if (base != new_base) {        if (likely(base->running_timer != timer)) {            /* See the comment in lock_timer_base() */            timer_set_base(timer, NULL);            spin_unlock(&base->lock);            base = new_base;            spin_lock(&base->lock);            timer_set_base(timer, base);        }    }    timer->expires = expires;    internal_add_timer(base, timer);    spin_unlock_irqrestore(&base->lock, flags);    return ret;}

  代码解释:

  注:卸载软件时钟的意思是指将软件时钟从软件时钟所在 base 中删除,以后所说的卸载软件时钟也都是这个意思

  取得软件时钟所在 base 上的同步锁( struct tvec_base 变量中的自旋锁),并返回该软件时钟的 base ,保存在 base 变量中

  如果该软件时钟处在 pending 状态(在 base 中,准备执行),则卸载该软件时钟

  取得本 CPU 上的 base 指针(类型为 struct tvec_base* ),保存在 new_base 中

  如果 base 和 new_base 不一样,也就是说软件时钟发生了迁移(从一个 CPU 中移到了另一个 CPU 上),那么如果该软件时钟的处理函数当前没有在迁移之前的那个 CPU 上运行,则先将软件时钟的 base 设置为 NULL ,然后再将该软件时钟的 base 设置为 new_base 。否则,跳到5。

  设置软件时钟的到期时间

  调用 internal_add_timer 函数将软件时钟添加到软件时钟的 base 中(本 CPU 的 base )

  释放锁

  这里有必要详细说明一下软件时钟如何被添加到软件时钟的 base 中的(添加到本 CPU base 的 tv1~tv5 里面),因为这是软件时钟处理的基础。来看函数 internal_add_timer 函数的实现,如清单3-3

  清单3-3 internal_add_timer 函数

  static void internal_add_timer(struct tvec_base *base, struct timer_list *timer){    unsigned long expires = timer->expires;    unsigned long idx = expires - base->timer_jiffies;    struct list_head *vec;    if (idx tv1.vec + i;    } else if (idx <1 <<(TVR_BITS + TVN_BITS)) {        int i = (expires >> TVR_BITS) & TVN_MASK;        vec = base->tv2.vec + i;    } else if (idx <1 <<(TVR_BITS + 2 * TVN_BITS)) {        int i = (expires >> (TVR_BITS + TVN_BITS)) & TVN_MASK;        vec = base->tv3.vec + i;    } else if (idx <1 <<(TVR_BITS + 3 * TVN_BITS)) {        int i = (expires >> (TVR_BITS + 2 * TVN_BITS)) & TVN_MASK;        vec = base->tv4.vec + i;    } else if ((signed long) idx <0) {        vec = base->tv1.vec + (base->timer_jiffies & TVR_MASK);    } else {        int i;        if (idx > 0xffffffffUL) {            idx = 0xffffffffUL;            expires = idx + base->timer_jiffies;        }        i = (expires >> (TVR_BITS + 3 * TVN_BITS)) & TVN_MASK;        vec = base->tv5.vec + i;    }    list_add_tail(&timer->entry, vec);}

  代码解释:

  计算该软件时钟的到期时间和 timer_jiffies (当前正在处理的软件时钟的到期时间)的差值,作为索引保存到 idx 变量中。

  判断 idx 所在的区间,在

  [0, ]或者( , 0)(该软件时钟已经到期),则将要添加到 tv1 中

  [, ],则将要添加到 tv2 中

  [, ],则将要添加到 tv3 中

  [, ],则将要添加到 tv4 中

  [, ),则将要添加到 tv5 中,但实际上最大值为 0xffffffffUL

  计算所要加入的具体位置(哪个链表中,即 tv1~tv5 的哪个子链表,参考图3-1)

  最后将其添加到相应的链表中

  从这个函数可以得知,内核中是按照软件时钟到期时间的相对值(相对于 timer_jiffies 的值)将软件时钟添加到软件时钟所在的 base 中的。

  3.3.2 删除软件时钟

  内核可调用 del_timer 函数删除软件时钟, del_timer 的代码如清单3-4

  清单3-4 del_timer 函数

  int del_timer(struct timer_list *timer){    struct tvec_base *base;    unsigned long flags;    int ret = 0;    ……    if (timer_pending(timer)) {        base = lock_timer_base(timer, &flags);        if (timer_pending(timer)) {            detach_timer(timer, 1);            ret = 1;        }        spin_unlock_irqrestore(&base->lock, flags);    }    return ret;}

  代码解释:

  检测该软件时钟是否处在 pending 状态(在 base 中,准备运行),如果不是则直接函数返回

  如果处于 pending 状态,则获得锁

  再次检测软件时钟是否处于 pending 状态(该软件时钟可能被卸载了),不是则释放锁然后函数返回

  如果还是 pending 状态,则将其卸载,之后释放锁,函数返回

  如果在 SMP 系统中,则需使用 del_timer_sync 函数来删除软件时钟。在讲解 del_timer_sync 函数之前,先来看下 try_to_del_timer_sync 函数的实现(该函数被 del_timer_sync 函数使用),其代码如清单3-5

  清单3-5 try_to_del_timer_sync 函数

  int try_to_del_timer_sync(struct timer_list *timer){    struct tvec_base *base;    unsigned long flags;    int ret = -1;    base = lock_timer_base(timer, &flags);    if (base->running_timer == timer)        goto out;    ret = 0;    if (timer_pending(timer)) {        detach_timer(timer, 1);        ret = 1;    }out:    spin_unlock_irqrestore(&base->lock, flags);    return ret;}

  该函数检测当前运行的软件时钟是不是该软件时钟,如果是,则函数返回-1,表明目前不能删除该软件时钟;如果不是检测该软件时钟是否处于 pending 状态,如果不是,则函数返回0,表明软件时钟已经被卸载,如果处于 pending 状态再把软件时钟卸载,函数返回1,表明成功卸载该软件时钟。

[1] [2] [3] 下一页


推荐阅读
  • 在Docker中,将主机目录挂载到容器中作为volume使用时,常常会遇到文件权限问题。这是因为容器内外的UID不同所导致的。本文介绍了解决这个问题的方法,包括使用gosu和suexec工具以及在Dockerfile中配置volume的权限。通过这些方法,可以避免在使用Docker时出现无写权限的情况。 ... [详细]
  • 学习SLAM的女生,很酷
    本文介绍了学习SLAM的女生的故事,她们选择SLAM作为研究方向,面临各种学习挑战,但坚持不懈,最终获得成功。文章鼓励未来想走科研道路的女生勇敢追求自己的梦想,同时提到了一位正在英国攻读硕士学位的女生与SLAM结缘的经历。 ... [详细]
  • 本文介绍了数据库的存储结构及其重要性,强调了关系数据库范例中将逻辑存储与物理存储分开的必要性。通过逻辑结构和物理结构的分离,可以实现对物理存储的重新组织和数据库的迁移,而应用程序不会察觉到任何更改。文章还展示了Oracle数据库的逻辑结构和物理结构,并介绍了表空间的概念和作用。 ... [详细]
  • vue使用
    关键词: ... [详细]
  • 一、Hadoop来历Hadoop的思想来源于Google在做搜索引擎的时候出现一个很大的问题就是这么多网页我如何才能以最快的速度来搜索到,由于这个问题Google发明 ... [详细]
  • 生成式对抗网络模型综述摘要生成式对抗网络模型(GAN)是基于深度学习的一种强大的生成模型,可以应用于计算机视觉、自然语言处理、半监督学习等重要领域。生成式对抗网络 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文介绍了brain的意思、读音、翻译、用法、发音、词组、同反义词等内容,以及脑新东方在线英语词典的相关信息。还包括了brain的词汇搭配、形容词和名词的用法,以及与brain相关的短语和词组。此外,还介绍了与brain相关的医学术语和智囊团等相关内容。 ... [详细]
  • 云原生边缘计算之KubeEdge简介及功能特点
    本文介绍了云原生边缘计算中的KubeEdge系统,该系统是一个开源系统,用于将容器化应用程序编排功能扩展到Edge的主机。它基于Kubernetes构建,并为网络应用程序提供基础架构支持。同时,KubeEdge具有离线模式、基于Kubernetes的节点、群集、应用程序和设备管理、资源优化等特点。此外,KubeEdge还支持跨平台工作,在私有、公共和混合云中都可以运行。同时,KubeEdge还提供数据管理和数据分析管道引擎的支持。最后,本文还介绍了KubeEdge系统生成证书的方法。 ... [详细]
  • Echarts图表重复加载、axis重复多次请求问题解决记录
    文章目录1.需求描述2.问题描述正常状态:问题状态:3.解决方法1.需求描述使用Echats实现了一个中国地图:通过选择查询周期&#x ... [详细]
  • 本文介绍了设计师伊振华受邀参与沈阳市智慧城市运行管理中心项目的整体设计,并以数字赋能和创新驱动高质量发展的理念,建设了集成、智慧、高效的一体化城市综合管理平台,促进了城市的数字化转型。该中心被称为当代城市的智能心脏,为沈阳市的智慧城市建设做出了重要贡献。 ... [详细]
  • IhaveconfiguredanactionforaremotenotificationwhenitarrivestomyiOsapp.Iwanttwodiff ... [详细]
  • Python字典推导式及循环列表生成字典方法
    本文介绍了Python中使用字典推导式和循环列表生成字典的方法,包括通过循环列表生成相应的字典,并给出了执行结果。详细讲解了代码实现过程。 ... [详细]
  • 本文讨论了在Windows 8上安装gvim中插件时出现的错误加载问题。作者将EasyMotion插件放在了正确的位置,但加载时却出现了错误。作者提供了下载链接和之前放置插件的位置,并列出了出现的错误信息。 ... [详细]
  • “你永远都不知道明天和‘公司的意外’哪个先来。”疫情期间,这是我们最战战兢兢的心情。但是显然,有些人体会不了。这份行业数据,让笔者“柠檬” ... [详细]
author-avatar
艾灸养生加盟
艾灸加盟热线:037761297867 15638966697(王经理) 18237761726(周经理)
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有