在Linux下写一个驱动时候遇到的读操作性能问题,让我想一窥系统调用的处理流程,以查出问题的root cause。很多书把它和中断处理放在一起讲,但是又没有哪本书说清楚了,看来只有代码才能说明一切。以Linux系统下MIPS体系结构为例。

从开始说起。

1. 相关代码

 trap_init(void) /* 系统初始化:start_kernel中 */
set_handler(0x180, &except_vec3_generic, 0x80);/* except_vec3_generic 根据cause寄存器跳转到其若干类异常/中断处理函数中*/
set_except_vector(, rollback ? rollback_handle_int : handle_int);
set_except_vector(, handle_tlbm);
... ...
set_except_vector(, handle_sys);

当系统发现异常时,CPU将自动进入内核模式并禁止中断,同时将PC指针指向默认的地址(根据异常的不同将分别进入偏移地址为0x180 or 0x200的地方,这些在函数trap_init都有体现,上述代码没有一一贴出)。

异常号为8时进入系统调用的处理入口。

 /* 进入系统调用异常处理 */
handle_sys
SAVE_SOME
STI # 进入内核模式并enable中断
la t1, sys_call_table
... ...

注意这里系统调用入口一开始就会enable中断。

而异常号为0时进入中断处理的入口。

 /* 进入中断异常处理 */
handle_int
SAVE_ALL
CLI # 进入内核模式并disable中断
   plat_irq_dispatch # 每种类型的CPU都有自己的实现
do_IRQ
generic_handle_irq /*处理用户注册的ISR,关中断进行 */
irq_exit()
do_softirq /*进入下半部处理(软中断) */
local_irq_enable /*仍然在中断上下文中,但是开中断*/
... ... /*处理用户注册的下半部,
Note:有可能在softirq内核线程中进行处理,所以下半部的处理并不总是在中断上下文中,但是下半部的思想就是允许中断嵌套*/

上面的代码描述了中断处理的框架,简要列出了处理中值得注意的地方。这里尤其注意中断处理入口一开始直接disable中断。

2. 结论

所以,系统调用和普通中断处理的异同在于:两者都是从同一个异常处理入口开始,但是系统调用会一开始让CPU进入内核模式且使能中断,然后从系统调用表中取得相应的注册函数调用之;而中断处理则让CPU进入内核模式且disable中断,继而调用do_IRQ函数。所以系统调用的真实处理(系统调用表中的注册函数执行)中可以阻塞,而中断处理的上半部不可以。所以在写驱动代码如字符设备驱动,实现读操作时是可以让其sleep的(比如没有数据时候,用户设置读模式是阻塞型的)。另一方面,如果该驱动读操作过于耗时也是不可取的,它在内核态中执行,这个时候只有中断的优先级比它高,其它的高优先级线程将不能得到及时调度执行。

另外,思考一下为什么在中断的下半部如tasklet中不能做阻塞的操作,而系统调用却可以?

首先,系统调用过程中阻塞意味着阻塞时(内核模式中)当前上下文放在当前线程栈中,同时系统调用所在线程状态改变(比如调用wait_event),下次改线程再次被调度后就可以从阻塞点继续run,这整个过程和一个普通线程阻塞并再次调度的过程是一样的,唯一的区别在于系统调用的阻塞点是在内核模式下。而下半部则不一样,作为中断处理的一个例程逻辑上不属于某个线程,所以它阻塞时第一不能改变当前线程的状态(而wait_semaphore之类的函数则会),其次它在阻塞的情况下再次run的点是在被中断线程再次被调度后,这也是不合理的,因为它不是该线程的一部分,凭什么让它的运行时间受限于该线程的调度时机?

系统调用和中断处理的异同(以Linux MIPS为例)的更多相关文章

  1. Prometheus Node_exporter 之 Network Netstat TCP Linux MIPs

    Network Netstat TCP Linux MIPs1. TCP Aborts / Tiemouts type: GraphUnit: shortLabel: ConnectionsTCPAb ...

  2. 【原创】xenomai内核解析--双核系统调用(二)--应用如何区分xenomai/linux系统调用或服务

    版权声明:本文为本文为博主原创文章,转载请注明出处.如有错误,欢迎指正. 1. 引出问题 上一篇文章xenomai内核解析--双核系统调用(一)以X86处理器为例,分析了xenomai内核调用的流程, ...

  3. linux 命令小例

    xargs示例: ls |xargs -i mv {}  /opt find示例: find -mtime +n -name “*.avi” -type f -exec rm {} \; find - ...

  4. linux目录结构详解(以suse linux 10为例)

    一.文件系统结构 位于Linux系统的最顶端即根目录是/.Linux的文件系统的入口就是/,所有的目录.文件.设备都在/之下,/就是Linux文件系统的组织者,也是最上级的领导者. 它之下的子目录有: ...

  5. [转]Linux学习笔记——例说makefile 头文件查找路径

    0.前言     从学习C语言开始就慢慢开始接触makefile,查阅了很多的makefile的资料但总感觉没有真正掌握makefile,如果自己动手写一个makefile总觉得非常吃力.所以特意借助 ...

  6. 命令行刷机教程( 以Linux系统为例 )

    //第一步adb device // 如果不能cd AndroidSDK/platform-toolsadb kill-server adb start-server //第二步adb reboot ...

  7. Linux学习笔记——例说makefile 头文件查找路径

    0.前言     从学习C语言開始就慢慢開始接触makefile,查阅了非常多的makefile的资料但总感觉没有真正掌握makefile,假设自己动手写一个makefile总认为非常吃力.所以特意借 ...

  8. Linux curl 一例

    root@PC-RENGUOQIANG:~# curl http://kermit:kermit@192.168.66.182:8080/activiti-rest/service/repositor ...

  9. Linux学习笔记——例说makefile 综合案例

    0.前言     从学习C语言開始就慢慢開始接触makefile,查阅了非常多的makefile的资料但总感觉没有真正掌握makefile,假设自己动手写一个makefile总认为非常吃力.所以特意借 ...

随机推荐

  1. ThinkPHP HTML标签代码和UBB互相转换

    1.UBB 转为 HTML TP的扩展里面自带一个ubb方法,用这个方法就能把用户输入的ubb格式代码转换为HTML标签的代码.这里用到的基本知识就是正则表达式啦,今天先不讲正则表达式. 来看一下TP ...

  2. PHP扩展Redis编译安装

    PHP扩展Redis编译安装 1.下载PHP官方Redis源码包  wget http://pecl.php.net/get/redis-2.2.4.tgz  注:我用的是Redhat系统,ubunt ...

  3. 那些年被我坑过的Python——邂逅与初识(第一章)

    第一问:为什么学习Python? 虚妖说:为了还债,还技术债,很早接触编程,却一直徘徊,也码了很多代码,却从未真真学会编程! 第二问:什么是Python 是一种以简洁.优雅著称的解释型.动态.强类型的 ...

  4. typedef 类型重命名 和 #define 宏定义(1)

    http://www.blogjava.net/jasmine214--love/archive/2010/11/29/339307.html 在现实生活中,信息的概念可能是长度,数量和面积等.在C语 ...

  5. Swift(一,创建对象,类型推导,基本运算,逻辑,字符串,数组,字典)

    swift用起来刚开始感觉有点怪怪的,但用了一段时间觉得还是挺好用的,哈哈.毕竟都是要有一个过程的嘛. 我就写一些自己在使用swift的时候的注意点吧,如有不正之处,还请指正! 一.在开发中优先使用常 ...

  6. 趣味C程序100.1 .2 绘制正弦曲线

    说明:1.本问题来源于<C语言经典.趣味.实用程序设计编程百例精解>,所有程序为本人自己编写.与原程序不同之处作有标记. 2.本系列所有程序均使用codeblocks编译,操作系统为Win ...

  7. 30+最佳Ajax jQuery的自动完成插件的例子

    在这篇文章中,我们将介绍35个jQuery AJAX的自动完成提示例子. jQuery 的自动完成功能,使用户快速找到并选择一定的价值.每个人都想要快速和即时搜索输入栏位,因为这个原因,许 流行的搜索 ...

  8. andorid studio

    http://www.cnblogs.com/smyhvae/p/4390905.html

  9. xstream 别名的用法<转>

    1.xstream的alias使用方法: 1.1 作用:将序列化中的类全量名称,用别名替换. 1.2  使用方法:xstream.alias("blog", Blog.class) ...

  10. 【uva10917】Walk Through the Forest (最短路)

    题目: gbn最近打算穿过一个森林,但是他比较傲娇,于是他决定只走一些特殊的道路,他打算只沿着满足如下条件的(A,B)道路走:存在一条从B出发回家的路,比所有从A出发回家的路径都短.你的任务是计算一共 ...