Linux中断(interrupt)子系统之一:arch相关的硬件封装层

linux的通用中断子系统的一个设计原则就是把底层的硬件实现尽可能地隐藏起来,使得驱动程序的开发人员不用关注底层的实现,要实现这个目标,内核的开发者们必须把硬件相关的内容剥离出来,然后定义一些列标准的接口供上层访问,上层的开发人员只要知道这些接口即可完成对中断的进一步处理和控制。对底层的封装主要包括两部分:
实现不同体系结构中断入口,这部分代码通常用asm实现;
中断控制器进行封装和实现;
本文的内容正是要讨论硬件封装层的实现细节。我将以arm体系进行介绍,大部分的代码位于内核代码树的arch/arm/目录内。
1. cpu的中断入口
我们知道,arm的异常和复位向量表有两种选择,一种是低端向量,向量地址位于0x00000000,另一种是高端向量,向量地址位于0xffff0000,linux选择使用高端向量模式,也就是说,当异常发生时,cpu会把pc指针自动跳转到始于0xffff0000开始的某一个地址上:
arm的异常向量表地址异常种类ffff0000复位ffff0004未定义指令ffff0008软中断(swi)ffff000cprefetch abortffff0010data abortffff0014保留ffff0018irqffff001cfiq
中断向量表在arch/arm/kernel/entry_armv.s中定义,为了方便讨论,下面只列出部分关键的代码:
[plain]view plaincopy
.globl__stubs_start
__stubs_start:
vector_stubirq,irq_mode,4
.long__irq_usr@0(usr_26/usr_32)
.long__irq_invalid@1(fiq_26/fiq_32)
.long__irq_invalid@2(irq_26/irq_32)
.long__irq_svc@3(svc_26/svc_32)
vector_stubdabt,abt_mode,8
.long__dabt_usr@0(usr_26/usr_32)
.long__dabt_invalid@1(fiq_26/fiq_32)
.long__dabt_invalid@2(irq_26/irq_32)
.long__dabt_svc@3(svc_26/svc_32)
vector_fiq:
disable_fiq
subspc,lr,#4
......
.globl__stubs_end
__stubs_end:
.equstubs_offset,__vectors_start+0x200-__stubs_start
.globl__vectors_start
__vectors_start:
arm(swisys_error0)
thumb(svc#0)
thumb(nop)
w(b)vector_und+stubs_offset
w(ldr)pc,.lcvswi+stubs_offset
w(b)vector_pabt+stubs_offset
w(b)vector_dabt+stubs_offset
w(b)vector_addrexcptn+stubs_offset
w(b)vector_irq+stubs_offset
w(b)vector_fiq+stubs_offset
.globl__vectors_end
__vectors_end:
代码被分为两部分:
第一部分是真正的向量跳转表,位于__vectors_start和__vectors_end之间;
第二部分是处理跳转的部分,位于__stubs_start和__stubs_end之间;
[plain]view plaincopy
vector_stubirq,irq_mode,4
以上这一句把宏展开后实际上就是定义了vector_irq,根据进入中断前的cpu模式,分别跳转到__irq_usr或__irq_svc。
[plain]view plaincopy
vector_stubdabt,abt_mode,8
以上这一句把宏展开后实际上就是定义了vector_dabt,根据进入中断前的cpu模式,分别跳转到__dabt_usr或__dabt_svc。
系统启动阶段,位于arch/arm/kernel/traps.c中的early_trap_init()被调用:
[cpp]view plaincopy
void__initearly_trap_init(void)
{
......
/*
*copythevectors,stubsandkuserhelpers(inentry-armv.s)
*intothevectorpage,mappedat0xffff0000,andensurethese
*arevisibletotheinstructionstream.
*/
memcpy((void*)vectors,__vectors_start,__vectors_end-__vectors_start);
memcpy((void*)vectors+0x200,__stubs_start,__stubs_end-__stubs_start);
......
}
以上两个memcpy会把__vectors_start开始的代码拷贝到0xffff0000处,把__stubs_start开始的代码拷贝到0xffff0000+0x200处,这样,异常中断到来时,cpu就可以正确地跳转到相应中断向量入口并执行他们。
图1.1 linux中arm体系的中断向量拷贝过程
对于系统的外部设备来说,通常都是使用irq中断,所以我们只关注__irq_usr和__irq_svc,两者的区别是进入和退出中断时是否进行用户栈和内核栈之间的切换,还有进程调度和抢占的处理等,这些细节不在这里讨论。两个函数最终都会进入irq_handler这个宏:
[plain]view plaincopy
.macroirq_handler
#ifdefconfig_multi_irq_handler
ldrr1,=handle_arch_irq
movr0,sp
adrlr,bsym(9997f)
ldrpc,[r1]
#else
arch_irq_handler_default
#endif
9997:
.endm
如果选择了multi_irq_handler配置项,则意味着允许平台的代码可以动态设置irq处理程序,平台代码可以修改全局变量:handle_arch_irq,从而可以修改irq的处理程序。这里我们讨论默认的实现:arch_irq_handler_default,它位于arch/arm/include/asm/entry_macro_multi.s中:
[plain]view plaincopy
.macroarch_irq_handler_default
get_irqnr_preambler6,lr
1:get_irqnr_and_baser0,r2,r6,lr
movner1,sp
@
@routinecalledwithr0=irqnumber,r1=structpt_regs*
@
adrnelr,bsym(1b)
bneasm_do_irq
......
get_irqnr_preamble和get_irqnr_and_base两个宏由machine级的代码定义,目的就是从中断控制器中获得irq编号,紧接着就调用asm_do_irq,从这个函数开始,中断程序进入c代码中,传入的参数是irq编号和寄存器结构指针,这个函数在arch/arm/kernel/irq.c中实现:
[cpp]view plaincopy
/*
*asm_do_irqistheinterfacetobeusedfromassemblycode.
*/
asmlinkagevoid__exception_irq_entry
asm_do_irq(unsignedintirq,structpt_regs*regs)
{
handle_irq(irq,regs);
}
到这里,中断程序完成了从asm代码到c代码的传递,并且获得了引起中断的irq编号。
2. 初始化
与通用中断子系统相关的初始化由start_kernel()函数发起,调用流程如下图所视:
图2.1 通用中断子系统的初始化
首先,在setup_arch函数中,early_trap_init被调用,其中完成了第1节所说的中断向量的拷贝和重定位工作。
然后,start_kernel发出early_irq_init调用,early_irq_init属于与硬件和平台无关的通用逻辑层,它完成irq_desc结构的内存申请,为它们其中某些字段填充默认值,完成后调用体系相关的arch_early_irq_init函数完成进一步的初始化工作,不过arm体系没有实现arch_early_irq_init。
接着,start_kernel发出init_irq调用,它会直接调用所属板子machine_desc结构体中的init_irq回调。machine_desc通常在板子的特定代码中,使用machine_start和machine_end宏进行定义。
machine_desc->init_irq()完成对中断控制器的初始化,为每个irq_desc结构安装合适的流控handler,为每个irq_desc结构安装irq_chip指针,使他指向正确的中断控制器所对应的irq_chip结构的实例,同时,如果该平台中的中断线有多路复用(多个中断公用一个irq中断线)的情况,还应该初始化irq_desc中相应的字段和标志,以便实现中断控制器的级联。
3. 中断控制器的软件抽象:struct irq_chip
正如上一篇文章linux中断(interrupt)子系统之一:中断系统基本原理所述,所有的硬件中断在到达cpu之前,都要先经过中断控制器进行汇集,合乎要求的中断请求才会通知cpu进行处理,中断控制器主要完成以下这些功能:
对各个irq的优先级进行控制;
向cpu发出中断请求后,提供某种机制让cpu获得实际的中断源(irq编号);
控制各个irq的电气触发条件,例如边缘触发或者是电平触发;
使能(enable)或者屏蔽(mask)某一个irq;
提供嵌套中断请求的能力;
提供清除中断请求的机制(ack);
有些控制器还需要cpu在处理完irq后对控制器发出eoi指令(end of interrupt);
在smp系统中,控制各个irq与cpu之间的亲缘关系(affinity);
通用中断子系统把中断控制器抽象为一个数据结构:struct irq_chip,其中定义了一系列的操作函数,大部分多对应于上面所列的某个功能:
[cpp]view plaincopy
structirq_chip{
constchar*name;
unsignedint(*irq_startup)(structirq_data*data);
void(*irq_shutdown)(structirq_data*data);
void(*irq_enable)(structirq_data*data);
void(*irq_disable)(structirq_data*data);
void(*irq_ack)(structirq_data*data);
void(*irq_mask)(structirq_data*data);
void(*irq_mask_ack)(structirq_data*data);
void(*irq_unmask)(structirq_data*data);
void(*irq_eoi)(structirq_data*data);
int(*irq_set_affinity)(structirq_data*data,conststructcpumask*dest,boolforce);
int(*irq_retrigger)(structirq_data*data);
int(*irq_set_type)(structirq_data*data,unsignedintflow_type);
int(*irq_set_wake)(structirq_data*data,unsignedinton);
void(*irq_bus_lock)(structirq_data*data);
void(*irq_bus_sync_unlock)(structirq_data*data);
void(*irq_cpu_online)(structirq_data*data);
void(*irq_cpu_offline)(structirq_data*data);
void(*irq_suspend)(structirq_data*data);
void(*irq_resume)(structirq_data*data);
void(*irq_pm_shutdown)(structirq_data*data);
void(*irq_print_chip)(structirq_data*data,structseq_file*p);
unsignedlongflags;
/*currentlyusedonlybyuml,mightdisappearoneday.*/
#ifdefconfig_irq_release_method
void(*release)(unsignedintirq,void*dev_id);
#endif
};
各个字段解释如下:
name中断控制器的名字,会出现在 /proc/interrupts中。
irq_startup 第一次开启一个irq时使用。
irq_shutdown 与irq_starup相对应。
irq_enable 使能该irq,通常是直接调用irq_unmask()。
irq_disable 禁止该irq,通常是直接调用irq_mask,严格意义上,他俩其实代表不同的意义,disable表示中断控制器根本就不响应该irq,而mask时,中断控制器可能响应该irq,只是不通知cpu,这时,该irq处于pending状态。类似的区别也适用于enable和unmask。
irq_ack 用于cpu对该irq的回应,通常表示cpu希望要清除该irq的pending状态,准备接受下一个irq请求。
irq_mask 屏蔽该irq。
irq_unmask 取消屏蔽该irq。
irq_mask_ack 相当于irq_mask + irq_ack。
irq_eoi 有些中断控制器需要在cpu处理完该irq后发出eoi信号,该回调就是用于这个目的。
irq_set_affinity 用于设置该irq和cpu之间的亲缘关系,就是通知中断控制器,该irq发生时,那些cpu有权响应该irq。当然,中断控制器会在软件的配合下,最终只会让一个cpu处理本次请求。
irq_set_type 设置irq的电气触发条件,例如irq_type_level_high或irq_type_edge_rising。
irq_set_wake 通知电源管理子系统,该irq是否可以用作系统的唤醒源。
以上大部分的函数接口的参数都是irq_data结构指针,irq_data结构的由来在上一篇文章已经说过,这里仅贴出它的定义,各字段的意义请参考注释:
[cpp]view plaincopy
/**
*structirq_data-perirqandirqchipdatapasseddowntochipfunctions
*@irq:interruptnumber
*@hwirq:hardwareinterruptnumber,localtotheinterruptdomain
*@node:nodeindexusefulforbalancing
*@state_use_accessors:statusinformationforirqchipfunctions.
*useaccessorfunctionstodealwithit
*@chip:lowlevelinterrupthardwareaccess
*@domain:interrupttranslationdomain;responsibleformapping
*betweenhwirqnumberandlinuxirqnumber.
*@handler_data:per-irqdatafortheirq_chipmethods
*@chip_data:platform-specificper-chipprivatedataforthechip
*methods,toallowsharedchipimplementations
*@msi_desc:msidescriptor
*@affinity:irqaffinityonsmp
*
*thefieldshereneedtooverlaytheonesinirq_descuntilwe
*cleanedupthedirectreferencesandswitchedeverythingoverto
*irq_data.
*/
structirq_data{
unsignedintirq;
unsignedlonghwirq;
unsignedintnode;
unsignedintstate_use_accessors;
structirq_chip*chip;
structirq_domain*domain;
void*handler_data;
void*chip_data;
structmsi_desc*msi_desc;
#ifdefconfig_smp
cpumask_var_taffinity;
#endif
};
根据设备使用的中断控制器的类型,体系架构的底层的开发只要实现上述接口中的各个回调函数,然后把它们填充到irq_chip结构的实例中,最终把该irq_chip实例注册到irq_desc.irq_data.chip字段中,这样各个irq和中断控制器就进行了关联,只要知道irq编号,即可得到对应到irq_desc结构,进而可以通过chip指针访问中断控制器。
4. 进入流控处理层
进入c代码的第一个函数是asm_do_irq,在arm体系中,这个函数只是简单地调用handle_irq:
[cpp]view plaincopy
/*
*asm_do_irqistheinterfacetobeusedfromassemblycode.
*/
asmlinkagevoid__exception_irq_entry
asm_do_irq(unsignedintirq,structpt_regs*regs)
{
handle_irq(irq,regs);
}
handle_irq本身也不是很复杂:
[cpp]view plaincopy
voidhandle_irq(unsignedintirq,structpt_regs*regs)
{
structpt_regs*old_regs=set_irq_regs(regs);
irq_enter();
/*
*somehardwaregivesrandomlywronginterrupts.rather
*thancrashing,dosomethingsensible.
*/
if(unlikely(irq>=nr_irqs)){
if(printk_ratelimit())
printk(kern_warningbadirq%u,irq);
ack_bad_irq(irq);
}else{
generic_handle_irq(irq);
}
/*at91specificworkaround*/
irq_finish(irq);
irq_exit();
set_irq_regs(old_regs);
}
irq_enter主要是更新一些系统的统计信息,同时在__irq_enter宏中禁止了进程的抢占:
[cpp]view plaincopy
#define__irq_enter()
do{
account_system_vtime(current);
add_preempt_count(hardirq_offset);
trace_hardirq_enter();
}while(0)
cpu一旦响应irq中断后,arm会自动把cpsr中的i位置位,表明禁止新的irq请求,直到中断控制转到相应的流控层后才通过local_irq_enable()打开。你可能会奇怪,既然此时的irq中断都是都是被禁止的,为何还要禁止抢占?这是因为要考虑中断嵌套的问题,一旦流控层或驱动程序主动通过local_irq_enable打开了irq,而此时该中断还没处理完成,新的irq请求到达,这时代码会再次进入irq_enter,在本次嵌套中断返回时,内核不希望进行抢占调度,而是要等到最外层的中断处理完成后才做出调度动作,所以才有了禁止抢占这一处理。
下一步,generic_handle_irq被调用,generic_handle_irq是通用逻辑层提供的api,通过该api,中断的控制被传递到了与体系结构无关的中断流控层:
[cpp]view plaincopy
intgeneric_handle_irq(unsignedintirq)
{
structirq_desc*desc=irq_to_desc(irq);
if(!desc)
return-einval;
generic_handle_irq_desc(irq,desc);
return0;
}
最终会进入该irq注册的流控处理回调中:
[cpp]view plaincopy
staticinlinevoidgeneric_handle_irq_desc(unsignedintirq,structirq_desc*desc)
{
desc->handle_irq(irq,desc);
}
5. 中断控制器的级联
在实际的设备中,经常存在多个中断控制器,有时多个中断控制器还会进行所谓的级联。为了方便讨论,我们把直接和cpu相连的中断控制器叫做根控制器,另外一些和跟控制器相连的叫子控制器。根据子控制器的位置,我们把它们分为两种类型:
机器级别的级联 子控制器位于soc内部,或者子控制器在soc的外部,但是是某个板子系列的标准配置,如图5.1的左边所示;
设备级别的级联 子控制器位于某个外部设备中,用于汇集该设备发出的多个中断,如图5.1的右边所示;
图5.1 中断控制器的级联类型
对于机器级别的级联,级联的初始化代码理所当然地位于板子的初始化代码中(arch/xxx/mach-xxx),因为只要是使用这个板子或soc的设备,必然要使用这个子控制器。而对于设备级别的级联,因为该设备并不一定是系统的标配设备,所以中断控制器的级联操作应该在该设备的驱动程序中实现。机器设备的级联,因为得益于事先已经知道子控制器的硬件连接信息,内核可以方便地为子控制器保留相应的irq_desc结构和irq编号,处理起来相对简单。设备级别的级联则不一样,驱动程序必须动态地决定组合设备中各个子设备的irq编号和irq_desc结构。本章我只讨论机器级别的级联,设备级别的关联可以使用同样的原理,也可以实现为共享中断,我会在本系列接下来的文章中讨论。
要实现中断控制器的级联,要使用以下几个的关键数据结构字段和通用中断逻辑层的api:
irq_desc.handle_irq irq的流控处理回调函数,子控制器在把多个irq汇集起来后,输出端连接到根控制器的其中一个irq中断线输入脚,这意味着,每个子控制器的中断发生时,cpu一开始只会得到根控制器的irq编号,然后进入该irq编号对应的irq_desc.handle_irq回调,该回调我们不能使用流控层定义好的几个流控函数,而是要自己实现一个函数,该函数负责从子控制器中获得irq的中断源,并计算出对应新的irq编号,然后调用新irq所对应的irq_desc.handle_irq回调,这个回调使用流控层的标准实现。
irq_set_chained_handler() 该api用于设置根控制器与子控制器相连的irq所对应的irq_desc.handle_irq回调函数,并且设置irq_noprobe和irq_nothread以及irq_norequest标志,这几个标志保证驱动程序不会错误地申请该irq,因为该irq已经被作为级联irq使用。
irq_set_chip_and_handler() 该api同时设置irq_desc中的handle_irq回调和irq_chip指针。
以下例子代码位于:/arch/arm/plat-s5p/irq-eint.c:
[cpp]view plaincopy
int__inits5p_init_irq_eint(void)
{
intirq;
for(irq=irq_eint(0);irq<=irq_eint(15);irq++)
irq_set_chip(irq,&s5p_irq_vic_eint);
for(irq=irq_eint(16);irq<=irq_eint(31);irq++){
irq_set_chip_and_handler(irq,&s5p_irq_eint,handle_level_irq);
set_irq_flags(irq,irqf_valid);
}
irq_set_chained_handler(irq_eint16_31,s5p_irq_demux_eint16_31);
return0;
}
该soc芯片的外部中断:irq_eint(0)到irq_eint(15),每个引脚对应一个根控制器的irq中断线,它们是正常的irq,无需级联。irq_eint(16)到irq_eint(31)经过子控制器汇集后,统一连接到根控制器编号为irq_eint16_31这个中断线上。可以看到,子控制器对应的irq_chip是s5p_irq_eint,子控制器的irq默认设置为电平中断的流控处理函数handle_level_irq,它们通过api:irq_set_chained_handler进行设置。如果根控制器有128个中断线,irq_eint0--irq_eint15通常占据128内的某段连续范围,这取决于实际的物理连接。irq_eint16_31因为也属于跟控制器,所以它的值也会位于128以内,但是irq_eint16--irq_eint31通常会在128以外的某段范围,这时,代表irq数量的常量nr_irqs,必须考虑这种情况,定义出超过128的某个足够的数值。级联的实现主要依靠编号为irq_eint16_31的流控处理程序:s5p_irq_demux_eint16_31,它的最终实现类似于以下代码:
[cpp]view plaincopy
staticinlinevoids5p_irq_demux_eint(unsignedintstart)
{
u32status=__raw_readl(s5p_eint_pend(eint_reg_nr(start)));
u32mask=__raw_readl(s5p_eint_mask(eint_reg_nr(start)));
unsignedintirq;
status&=~mask;
status&=0xff;
while(status){
irq=fls(status)-1;
generic_handle_irq(irq+start);
status&=~(1< }
}
在获得新的irq编号后,它的最关键的一句是调用了通用中断逻辑层的api:generic_handle_irq,这时它才真正地把中断控制权传递到中断流控层中来。

使用笔记本电脑的错误操作
虚拟现实与艺术相碰撞会擦出怎样的火花
什么是分辨率
一波未平一波又起:杭州出现共享汽车坟场
中控智慧科技面部门禁一体机TA1200简介
Linux中断(interrupt)子系统之一:arch相关的硬件封装层
柔性电路板适用范围
一加5手机什么时候上市?售价预计多少?
无人机告诉你,如何用编程制作“报数游戏”?
解决方案 | 基于全志T507核心板设计智能加油机应用
一个巧妙好玩儿的单键轻触电子开关电路
什么是串联谐振_串联谐振电路的特征_串联谐振和并联谐振的条件
基于WT2003H模组的雾化加湿器设计方案
中芯国际14nm能够发挥7nm工艺存疑
DRAM的变数
XTKB-982S开关柜状态指示器
人工智能如何改善你的睡眠
BUCK电路仿真建模案例
联发科发布2019年3月份及第1季营收资料 整体表现呈现淡季不淡的情况
如何保护以太网供电网络