我有一个简单的微型灯设置,带有两个Gpio(按钮和开关)。 我想处理两个设备中断。
以下是我目前的非工作代码。
#include<xparameters.h>
#include<xgpio.h>
#include<xintc.h>
#include<xil_exception.h>
static XGpio PushBt;
static XGpio sw;
static XIntc myIntc;
int delay, limit=3000000;
void pb_int_handler(void *baseaddr_p) {
Xuint32 dsr;
//DSR contains the INFORMATION of which button was depressed, so we can switch
//on its value.
dsr = XGpio_DiscreteRead(&PushBt, 1);
switch(dsr) {
case 0x01:
xil_printf("Up\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
break;
case 0x02:
xil_printf("Left\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
break;
case 0x08:
xil_printf("Right\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
break;
case 0x04:
xil_printf("Down\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
break;
default : {}
}
//Clear the interrupt both in the Gpio instance as well as the interrupt controller
XGpio_InterruptClear(&PushBt, 0x3);
XIntc_Acknowledge(&myIntc,XPAR_AXI_INTC_0_PUSH_IP2INTC_IRPT_INTR);
}
void sw_int_handler(void *baseaddr_p) {
Xuint32 dsr;
//DSR contains the INFORMATION of which button was depressed, so we can switch
//on its value.
dsr = XGpio_DiscreteRead(&sw, 1);
switch(dsr) {
case 0x00:
xil_printf("Switches 00 \r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
break;
case 0x01:
xil_printf("Switches 01 \r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
break;
case 0x02:
xil_printf("Switches 02 \r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
break;
case 0x03:
xil_printf("Switches 03 \r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
break;
default : {}
}
//Clear the interrupt both in the Gpio instance as well as the interrupt controller
XGpio_InterruptClear(&sw, 0x3);
XIntc_Acknowledge(&myIntc,XPAR_AXI_INTC_0_SW_IP2INTC_IRPT_INTR);
}
int main(void) {
xil_printf("Setting up peripherals...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
xil_printf("Setting up push buttons...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
XGpio_Initialize(&PushBt, XPAR_PUSH_DEVICE_ID);
XGpio_SetDataDirection(&PushBt,1,1); //set pb as input port
XGpio_InterruptEnable(&PushBt, XPAR_PUSH_IP2INTC_IRPT_MASK);
XGpio_InterruptGlobalEnable(&PushBt);
xil_printf("Setting up switches...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
XGpio_Initialize(&sw, XPAR_SW_DEVICE_ID);
XGpio_SetDataDirection(&sw,1,1); //set sw as input port
XGpio_InterruptEnable(&sw, XPAR_SW_IP2INTC_IRPT_MASK);
XGpio_InterruptGlobalEnable(&sw);
xil_printf("Setting up interrupt controller...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
XIntc_Initialize(&myIntc, XPAR_INTC_0_DEVICE_ID);
xil_printf("Register the interrupt...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
XIntc_Connect(&myIntc, XPAR_AXI_INTC_0_PUSH_IP2INTC_IRPT_INTR,
(XInterruptHandler)pb_int_handler,
&PushBt);
XIntc_Connect(&myIntc, XPAR_AXI_INTC_0_SW_IP2INTC_IRPT_INTR,
(XInterruptHandler)sw_int_handler,
&sw);
xil_printf("Enable individual interrupt...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
XIntc_EnableIntr(&myIntc,XPAR_PUSH_IP2INTC_IRPT_MASK | XPAR_SW_IP2INTC_IRPT_MASK);
xil_printf("Start the interrupt controller...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
XIntc_Start(&myIntc, XIN_REAL_MODE);
XIntc_MasterEnable(&myIntc);
microblaze_enable_interrupts();
xil_printf("Setting up exceptions...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
Xil_ExceptionInit();
Xil_ExceptionRegisterHandler(XIL_EXCEPTION_ID_M_AXI_I_EXCEPTION,
(XExceptionHandler)XIntc_InterruptHandler,
&myIntc);
Xil_ExceptionEnable();
while(1) {
xil_printf("Entering loop...\r\n");
for(delay=0;delay<limit;delay++){}; // delay cycle
}
}
任何帮助解释正确的中断代码序列?
那我忘记了什么?或者有什么不对? 感谢
答案 0 :(得分:0)
找到解决方案:更改这些行
XGpio_InterruptEnable(&PushBt, XPAR_PUSH_IP2INTC_IRPT_MASK);
到
XGpio_InterruptEnable(&PushBt, 0xff);
和
XGpio_InterruptEnable(&sw, XPAR_SW_IP2INTC_IRPT_MASK);
到
XGpio_InterruptEnable(&sw, 0xff);
所以现在代码有效,但我会调查原因:)这是一个中断掩码问题
答案 1 :(得分:0)
您正在将错误的值传递给Mask
函数的XGpio_InterruptEnable
参数。
XPAR_PUSH_IP2INTC_IRPT_MASK
和XPAR_SW_IP2INTC_IRPT_MASK
是中断控制器外设的中断屏蔽值,而不是GPIO外设的中断屏蔽值。
这是C:\Xilinx\14.7\ISE_DS\EDK\sw\XilinxProcessorIPLib\drivers\gpio_v3_01_a\src\xgpio_l.h
的摘录,为您显示了GPIO外设的正确掩码值:
/** @name Interrupt Status and Enable Register bitmaps and masks
*
* Bit definitions for the interrupt status register and interrupt enable
* registers.
* @{
*/
#define XGPIO_IR_MASK 0x3 /**< Mask of all bits */
#define XGPIO_IR_CH1_MASK 0x1 /**< Mask for the 1st channel */
#define XGPIO_IR_CH2_MASK 0x2 /**< Mask for the 2nd channel */
/*@}*/
因此,您可能应该更改行
XGpio_InterruptEnable(&PushBt, XPAR_PUSH_IP2INTC_IRPT_MASK);
XGpio_InterruptEnable(&sw, XPAR_SW_IP2INTC_IRPT_MASK);
到
XGpio_InterruptEnable(&PushBt, XGPIO_IR_CH1_MASK);
XGpio_InterruptEnable(&sw, XGPIO_IR_CH1_MASK);