在CPU级:
28x系列 CPU支持一个不可屏蔽中断NMI(non-maskable interrupt)和16个可屏蔽、可屏蔽中断(INT1-INT14、RTOSINT、DLOGINT)
(有些资料提到是2个不可屏蔽中断:RESET和NMI)
由于28x系列的许多外设都需要用到中断,因此仅仅这些中断通道显然是不够的,因此引入了PIE模块,管理外设中断
PIE模块主要管理的是INT1-INT12这12条线上的中断,每条线上分时复用8路中断,因此PIE模块管理者12乘8=96个中断源
F28335的外部中断并没有固定接口,其中GPIO0-GPIO31可配置为外部中断XINT1,XINT2和NMI,注意其中NMI不受PIE管理,而GPIO32-63可配置为XINT3至XINT7
软件编写时:
需要配置GpioIntRegs中的GPIOXINTnSEL寄存器来选择GPIO的端口,例如GpioIntRegs.GPIOXINT1SEL.bit.GPIOSEL = 0; //Xint1 is GPIO0
然后配置XIntruptRegs中的XINTnCR来配置中断的一些参数:例如:
// Configure XINT1
XIntruptRegs.XINT1CR.bit.POLARITY = 0; // Falling edge interrupt
// Enable XINT1
XIntruptRegs.XINT1CR.bit.ENABLE = 1; // Enable Xint1
其中XINTnCTR寄存器是用于计数的寄存器,可不配置
根据上述传输过程,如果是PIE模块管理的中断,还要进行PIE模块的配置,先简述PIE的原理:
如图所示:PIE管理的整个中断过程分为3级、外设级、PIE级、CPU级
整个中断处理过程如上所示:
(1)外设级
外设产生中断时,该中断事件相关的中断标志位(IF)置1。此时,如果该中断相应的中断使能寄存器(IE)也置为1,外设就会向PIE控制器发出一个中断请求。如果外设级中断没有被使能(相应的使能位为0),那么外设就不会向PIE发出中断请求,相应的中断标志位会一直保持置位状态,除非用软件清除。当然,在中断标志位保持在1的时候,一旦该中断被使能了,那么外设立马会向PIE发出中断申请。
注:不管在什么情况下,外设寄存器中的中断标志位必须采用软件进行清除。
小结:
外设中断的屏蔽,需要将与该中断相关的外设寄存中的中断使能位置0;
外设中断标志位的清除,需要将与该中断相关的外设寄存中的中断标志位 置1;
(2)PIE级
当外设产生中断事件,相关中断标志位置位,中断使能位使能之后,外设就会把中断请求提交给我们的PIE模块。PIE模块将96个外设和外部引脚的中断进行了分组,分为12组,每组8个中断,分别是PIE1-PIE12。每个组的中断被多路汇集进入1个CPU中断,例如PDPINDA,PDPINDB,XINT1,XINT2,ADCINT,TINT0,WAKEINT这7个中断都在PIE1组内,这些中断都汇集到CPU中断的INT1。 和外设级类似的,PIE控制器中的每个组都会有一个中断标志寄存器PIEIFRx和和中断使能寄存器PIEIERx,当然x=1.....12。每个寄存器的低8位对应于8个外设中断,高8位保留。
小结:
PIE中断的使能。就得将其相应组的使能寄存器PIEIERx的相应位进行置位;
PIE中断的屏蔽。这是和使能相反的操作;
PIE应答寄存器 PIEACK相关位的清除,以使得CPU能够响应同组的其他中断。
PIE级的中断和外设级的中断比较
外设中断的中断标志位是需要软件清除的,而PIE级的中断标志位都是自动置位或者清除的。但是PIE多了一个PIEACK寄存器,同一时间只能放一个中断过去,只有等到这个中断被响应,给PIEACK置位,才能让同组的下一个中断过去,被CPU响应,而PIEACK当响应一次中断后就会自动置位,因此需要软件写1清0
(3)CPU级
CPU也有标志寄存器IFR和使能寄存器IER。当某一个外设中断请求通过PIE发送到CPU时,CPU级中与INTx相关的中断标志位就会被置位。例如,T1的周期中断T1PINT的请求到达CPU这边时,与其相关的INT2的标志位就会被置位。这时候,该标志位就会被所存在IFR中,这时候,CPU不会马上去执行相应的中断,而是等待CPU使能IER寄存器的相关位,并且对CPU寄存器ST1中的全局中断屏蔽位做适当的使能。如果IER中的相关位被置位了,并且INTM的值为0,则中断就会被CPU响应。在T1PINT里,当IER的第2位即INT2被置位,INTM为0,则CPU就会响应定时器T1的周期中断。 CPU接到了终端的请求,就得暂停正在执行的程序,转而去响应中断程序,但是此时,它必须得做一些准备工作,以便于执行完中断程序之后回过头来还能找到原来的地方和原来的状态。CPU会将相应的IER和IFR位进行清除,EALLOW也被清除,INTM被置位,就是不能响应其他中断了,CPU向其他中断发出了通知,正在忙,没空来处理你们的请求了,得等到处理完手上的中断之后才能再来处理你们的请求。然后,CPU会存储返回地址并自动保存相关的信息,例如将正在处理的数据放入堆栈等等,做好这些准备工作之后,CPU会从PIE块中取出对应的中断向量ISR,从而转去执行中断子程序。
中断CPU级总结:
CPU级的操作都是自动的,不管是中断标志位(IFR),还是中断的使能位(IER)。
PIE96个中断对应的外设如下:
PIE级软件编程:
要对PIE寄存器进行操作,首先得使能PIE寄存器
PieCtrlRegs.PIECTRL.bit.ENPIE = 1; // Enable the PIE block
并且使能相应的PIE中断使能寄存器
PieCtrlRegs.PIEIER1.bit.INTx4 = 1; // Enable PIE Gropu 1 INT4 对应于外设中断向量表中的XINT1
PieCtrlRegs.PIEIER1.bit.INTx5 = 1; // Enable PIE Gropu 1 INT5对应于外设中断向量表中的XINT2
然后必须将对应的中断服务程序映射到中断向量表中
EALLOW;// This is needed to write to EALLOW protected registers
PieVectTable.XINT1 = &xint1_isr;
PieVectTable.XINT2 = &xint2_isr;
EDIS; // This is needed to disable write to EALLOW protected registers
CPU级软件编程:
IER |= M_INT1; // Enable CPU int1M_INT1对应0x0001使能对应的全局中断路径,也就是对应使能Group 1的中断
EINT; // Enable Global Interrupts 打开全局中断 对应汇编asm(" clrc INTM")
最后必须在中断服务程序中将对应Group1的PIEACKx=1写1清0
总的过程如下:
再提一下初始化时:
DINT; //关闭全局中断 对应汇编asm(" setc INTM")
InitPieCtrl(); //其中清除了所有PIE的PIEIERx与PIEIFRx
IER = 0x0000;
IFR = 0x0000;
清除CPU级的IER,IFR
InitPieVectTable(); //初始化中断服务程序,给每个中断源都赋了一个默认的中断处理程序