你好,一般使用中断都是写好普通中断函数,然后在中断初始化函数中将中断函数注册响应的中断号即可,例如下面这样:
/******************************************************************************* * @函数名称UART0_INT_Init * @函数说明串口0中断初始化函数 * @输入参数无 * @输出参数无 * @返回参数无 *******************************************************************************/ void UART0_INT_Init() {unsigned int intFlags = 0;// 使能中断intFlags |= (UART_INT_LINE_STAT | \UART_INT_TX_EMPTY |\UART_INT_RXDATA_CTI);UARTIntEnable(SOC_UART_0_REGS, intFlags);intFlags |= (UART_FCR_FIFOEN | \UART_FCR_DMAMODE1);UARTDMAEnable(SOC_UART_0_REGS, intFlags);// 禁用发送中断UARTIntDisable(SOC_UART_0_REGS, UART_INT_TX_EMPTY);CICDisableGlobalHostInt(SOC_CIC_0_REGS);CICEventMap(SOC_CIC_0_REGS, INTC0_UARTINT_A, uiCIC_out_num);CICEnableGlobalHostInt(SOC_CIC_0_REGS);IntRegister(C66X_MASK_INT6, UART0_Isr);IntEventMap(C66X_MASK_INT6, SYS_INT_CIC0_OUT0_20);IntEnable(C66X_MASK_INT6); }
平时都使用上面这种方式,没有什么问题。
但是最近看到一个例程,使用interrupt关键字定义,但是并未在其他地方进行中断注册,这种方式的实现原理是什么,在哪里有具体的讲解?代码如下:
interrupt void GE_Message_ISR() { EMAC_Desc *pDesc; int recv_bytes = 0; puts("Hello"); //EMAC_Recv((Uint8 *)0x10880000); /* check for new packet */ if(emac_mcb.lastrxhdp == (Uint32)EMAC_REGS->RX0HDP) //return 0;return; pDesc = (EMAC_Desc *)EMAC_REGS->RX0CP; if (pDesc->PktFlgLen & EMAC_DSC_FLAG_SOP) { /* Acknowledge recevied packet */ EMAC_REGS->RX0CP = (Uint32)pDesc; /* store bytes recevied */ recv_bytes = pDesc->PktFlgLen & 0xFFFF; /* optionally copy data to output buffer */ memcpy((Uint8 *)0x10880000, pDesc->pBuffer, recv_bytes); parserRxPacket(pDesc); /* re-initalize descriptor to recevie more data */ pDesc->BufOffLen = RX_BUFF_SIZE; pDesc->PktFlgLen = EMAC_DSC_FLAG_OWNER; /* assign descriptor to HDP */ if (0 == (Uint32)EMAC_REGS->RX0HDP) {EMAC_REGS->RX0HDP = (Uint32)pDesc; } else { ((EMAC_Desc *) (emac_mcb.lastrxhdp))->pNext = pDesc; } emac_mcb.lastrxhdp = (Uint32)pDesc; /* For debug purpose */ // printf("Receive packet length = %d,\n", recv_bytes); } EMAC_REGS->MACEOIVECTOR = 0x01; return; } void GE_Interrupts_Init(void) { /* Disable Global host interrupts. */ gpCIC0_regs->GLOBAL_ENABLE_HINT_REG= 0; /* Configure no nesting support in the CPINTC Module. */ gpCIC0_regs->CONTROL_REG= ((gpCIC0_regs->CONTROL_REG& ~CSL_CPINTC_CONTROL_REG_NEST_MODE_MASK)| (CPINTC_NO_NESTING<<CSL_CPINTC_CONTROL_REG_NEST_MODE_SHIFT)); /* Clear Interrupt events in MDIO*/ // gpMDIO_regs->LINK_INT_RAW_REG= 0xFFFFFFFF; /* Enable Global host interrupts. */ gpCIC0_regs->GLOBAL_ENABLE_HINT_REG= 1; /* Enable EMAC RX interrupts in the control module */ ECTL_REGS->C0_RX_EN = 0xFF; /*map EMAC RX interrupt to INT4 */ gpCGEM_regs->INTMUX1 = 99<<CSL_CGEM_INTMUX1_INTSEL4_SHIFT;/*Clear all DSP core events*/ gpCGEM_regs->EVTCLR[0]= 0xFFFFFFFF; gpCGEM_regs->EVTCLR[1]= 0xFFFFFFFF; gpCGEM_regs->EVTCLR[2]= 0xFFFFFFFF; gpCGEM_regs->EVTCLR[3]= 0xFFFFFFFF; //clear DSP core interrupt flag ICR= IFR; //enable INT4 and INT5 IER = 3|(1<<4)|(1<<5); /*Interrupt Service Table Pointer to begining of LL2 memory*/ ISTP= 0x800000; //enable GIE TSR = TSR|1; }
Shine:
interrupt关键字在编译器手册里有讲。
6.5.3 The interrupt Keyword
www.ti.com/…/spru187u.pdf
user4504841:
回复 Shine:
你好,我看了一下手册中interrupt关键字这段解释,但是稍微有点不明白。
例如上面这段代码中GE_Message_ISR被定义为中断函数,什么条件下这个中断会被触发,是中断初始函数GE_Interrupts_Init(void)中的//enable INT4 and INT5IER = 3|(1<<4)|(1<<5);
这句吗,所有的 INT4 and INT5均可以触发?
Shine:
回复 user4504841:
中断事件发生,中断会被触发,相应的中断标志位会被置位,如果这时总中断和相应的中断使能位使能而且是最高 优先级的话,就会跳转到中断子函数。
user4504841:
回复 Shine:
第一种中断使用方式很好理解,
CICDisableGlobalHostInt(SOC_CIC_0_REGS);
CICEventMap(SOC_CIC_0_REGS, INTC0_UARTINT_A, uiCIC_out_num);
CICEnableGlobalHostInt(SOC_CIC_0_REGS);
IntRegister(C66X_MASK_INT6, UART0_Isr);
IntEventMap(C66X_MASK_INT6, SYS_INT_CIC0_OUT0_20);
就是串口0中断标志位寄存器与中断号明确绑定注册,中断6和SYS_INT_CIC0_OUT0_20被置位时就会出发中断UART0_Isr。第二种中断定义方式,我在初始化函数中没法看到中断标志位、中断函数interrupt void GE_Message_ISR()与某个特定的中断号绑定注册,如何知道哪个中断标志位被置位时,会触发中断函数interrupt void GE_Message_ISR()?
user4504841:
回复 Shine:
非常感谢,是这样的,之前没注意到vectors.asm这个文件:
;-------------------------------------------------------------- ;interrupt vector table.sect "vecs".align 1024vectors:VEC_RESET _c_int00;RESETVEC_ENTRY NMI_ISR ;NMI/ExceptionVEC_DUMMY;RSVDVEC_DUMMY;RSVDVEC_ENTRY GE_Message_ISR;interrupt 4VEC_ENTRY GE_MISC_MDIO_ISR;interrupt 5VEC_DUMMY;interrupt 6VEC_DUMMY;interrupt 7VEC_DUMMY;interrupt 8VEC_DUMMY;interrupt 9VEC_DUMMY;interrupt 10VEC_DUMMY;interrupt 11VEC_DUMMY;interrupt 12VEC_DUMMY;interrupt 13VEC_DUMMY;interrupt 14VEC_DUMMY;interrupt 15.end这种中断注册方式使用起来没用第一种这么方便,什么情况下要用到这种方式,两者优缺点在哪里,在项目中两种方式同时使用要注意哪些问题呢?
Shine:
回复 user4504841:
在中断向量表里注册中断一般是在裸机程序中用。在BIOS程序中bios自带中断向量表,通过HWI配置即可。