Microblaze多个中断代码不起作用

时间:2014-10-25 19:16:30

标签: c interrupt microblaze

我有一个简单的微型灯设置,带有两个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
    }
}

任何帮助解释正确的中断代码序列?

  • 我设置了处理程序。
  • 我初始化了中断控制器。
  • 我连接了处理程序。
  • 我启用了中断屏蔽。
  • 我启动了中断控制器。
  • 我启用了主中断控制器。

那我忘记了什么?或者有什么不对? 感谢

2 个答案:

答案 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_MASKXPAR_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);