zoukankan      html  css  js  c++  java
  • 编写你的第一个GPIO 中断程序,ZEDBOARD, ZYNQ-7000

    Run you first gpio-ps interrupt application

    To run a gpio interrupt application, you should set the direction of specified pin and interrupt detect logic. Then route the gpio interrupt to scu gic module. And then route the scu gic to cpu exception table. This is the main plot.

    Configure GpioPs module

    1. Set the gpio pin as input and disable output enable switch
      • Init gpio ps module:
    XGpioPs GpioPs;
    
    XGpioPs_Config* pGpioPSCfg;
    
    pGpioPsCfg=XGpioPs_LookupConfig(XPAR_XGPIOPS_0_DEVICE_ID);
    
    XGpioPs_CfgInitialize(&GpioPs,pGpioPsCfg,pGpioPsCfg->BaseAddr);
      • Disable Output(actually, gpio pin input is always enabled and what you need to do is just disable the output)
    XGpioPs_SetOutputEnablePin(&GpioPs,50,0x0) ; //0 to disable output.
      • Set interrupt detect logic
    XGpioPs_SetIntrTypePin(&GpioPs,50,XGPIOPS_IRQ_EDGE_RISING);
    1. Set interrupt handler
      • Create a interrupt handler
    voidGpioPsHandler(void *CallBackRef, int Bank, u32 Status)
    
    {
    
        XGpioPs* pGpioPs=(XGpioPs*)CallBackRef;
    
        pGpioPs->IntrDisablePin(pGpioPs,50);
    
         xil_printf("GpioPs Handler running…");
    
        pGpioPs->IntrEnablePin(pGpioPs,50);
    
    }
      • Connect the handler to gpio module
    XGpioPs_SetCallbackHandler(&GpioPs,(void*)&GpioPs,GpioHandler);

    As we maybe need to use the function of gpio, so we need to use the instance: GpioPs , that is, transform the parameter as a void pointer:

    XGpioPs* pGpioPs=(XGpioPs*)CallBackRef;

      • Enable Gpio interrupt
    XGpioPS_IntrEnablePin(&GpioPs,50);

    Configure Scu Gic module

    The interrupt ID of gpiops module is #52, so 52 stands for the gpio interrupt.

    1. Init a SCU GIC instance
      XScuGic ScuGic;
      
      XScuGic_Config* pScuGicCfg;
      
      pScuGicCfg=XScuGic_LookupConfig(XPAR_SCUGIC_SINGLE_DEVICE_ID);
      
      XScuGic_CfgInitialize(&ScuGic,pScuGicCfg,pScuGicCfg->CpuBaseAddress);
      
      XScuGic_Disable(&ScuGic,52);
    2. Set the interrupt trigger type

      NOTICE: as the each interrupt route to gic module has a fixed interrupt signal, so you need to specifies the trigger types of scu gic module.

      The trigger type is different from the gpio interrupt trigger type. The gpio interrupt trigger type is user-specified , but the interrupt signal route to scu gic module is a fixed trigger type. Refer to ug585-zynq-7000-TRM.pdf page 189 for more details:

      XScuGic_SetPriorityTriggerType(&ScuGic,52,0xa0,0x01);

      The 3th parameter range from 0 to 0xFF, the less , the prior of the interrupt.

      The 4th parameter as shown:

    Software interrupt

    Read only

    0x02

    Private peripheral interrupt

    Read only

    B01 for high level

    B11 for rising edge

    Shared peripheral interrupt

    LSB Read only

    B01 for high level

    B11 for rising edge

     

    1. Connect the GpioPs handler to Exception table
      XScuGic_Connect(&ScuGic,52,(Xil_ExceptionHandler)XGpioPs_IntrHandler,(void*)&GpioPs);
    2. Enable the #52 Interupt in Scu Gic
      XScuGic_Enable(&ScuGic,52);

    Configure the Exception Table

    1. Init the Exception instance
      Xil_ExceptionInit();
    2. Connect the ScuGic to exception table
      Xil_ExceptionRegisterHandler(XIL_EXCEPTION_ID_IRQ_INT,(Xil_ExceptionHandler)XScuGic_IntrruptHandler,&ScuGic);
    3. Enable the XIL_EXCEPTION_ID_IRQ_INT exception
      Xil_ExceptionEnable();

      Now press the Button BTN8 on the Zedboard, just enjoy the message from console!

      Remember to configure the STDIO Connection. Otherwise , nothing print as you press.

      This is the exciting console print message:

    Appendix:

    Whole main.c contents:

    #include<stdio.h>

    #include"platform.h"

    #include"xgpiops.h"

    #include"xil_exception.h"

    #include"xscugic.h"

    void GpioPsHandler(void *CallBackRef, int Bank, u32 Status)

    {

        XGpioPs* pGpioPs=(XGpioPs*)CallBackRef;

        XGpioPs_IntrDisablePin(pGpioPs,50);

        xil_printf("GpioPs Handler running…");

        XGpioPs_IntrEnablePin(pGpioPs,50);

    }

    int main()

    {

    init_platform();

    XGpioPs GpioPs;

    XGpioPs_Config* pGpioPsCfg;

    pGpioPsCfg=XGpioPs_LookupConfig(XPAR_XGPIOPS_0_DEVICE_ID);

    XGpioPs_CfgInitialize(&GpioPs,pGpioPsCfg,pGpioPsCfg->BaseAddr);

    XGpioPs_SetOutputEnablePin(&GpioPs,50,0x0) ;

    XGpioPs_SetIntrTypePin(&GpioPs,50,XGPIOPS_IRQ_TYPE_EDGE_RISING);

    XGpioPs_SetCallbackHandler(&GpioPs,(void*)&GpioPs,GpioPsHandler);

    XGpioPs_IntrEnablePin(&GpioPs,50);

    XScuGic ScuGic;

    XScuGic_Config* pScuGicCfg;

    pScuGicCfg=XScuGic_LookupConfig(XPAR_SCUGIC_SINGLE_DEVICE_ID);

    XScuGic_CfgInitialize(&ScuGic,pScuGicCfg,pScuGicCfg->CpuBaseAddress);

    XScuGic_Disable(&ScuGic,52);

    XScuGic_SetPriorityTriggerType(&ScuGic,52,0xa0,0x01);

    XScuGic_Connect(&ScuGic,52,(Xil_ExceptionHandler)XGpioPs_IntrHandler,(void*)&GpioPs);

    XScuGic_Enable(&ScuGic,52);

    Xil_ExceptionInit();

    Xil_ExceptionRegisterHandler(XIL_EXCEPTION_ID_IRQ_INT,(Xil_ExceptionHandler)XScuGic_InterruptHandler,&ScuGic);

    Xil_ExceptionEnable();

    xil_printf("Gpio interrupt test, press BTN8 on zedboard. ");

    while(1)

    {

    u32 data;

            XGpioPs_SetOutputEnablePin(&GpioPs,0x10,0x0);

            data=XGpioPs_ReadPin(&GpioPs,0x10);

            XGpioPs_SetIntrTypePin(&GpioPs,0x10,XGPIOPS_IRQ_TYPE_EDGE_FALLING);

            data=XGpioPs_IntrGetStatus(&GpioPs,1);

            XGpioPs_IntrClear(&GpioPs,1,0xFFFFFFFF);

            data=XGpioPs_IntrGetStatus(&GpioPs,1);

            data=XGpioPs_ReadPin(&GpioPs,0x10);

    }

    return 0;

    }

    NOTICE:

    I have type some code in the main loop while(1), you can debug into the loop to see the value change of data.

    You will notice a existing interrupt pending on the pin 5th. No matter what you configure the pin , the interrupt always occurs.

    Why?

  • 相关阅读:
    iptables作为路由转发至后端web服务器
    AWS EC磁盘扩容针对XFS
    MongoDB集群分片部署
    理解Java对象:要从内存布局及底层机制说起,话说….
    类和对象在JVM中是如何存储的,竟然有一半人回答不上来!
    SpringBoot项目中,如何更规范的使用PageHelper分页?
    面试被问,一千万个整数里面快速查找某个整数,你会怎么去做?
    Spock单元测试框架实战指南三-If esle 多分支场景测试
    IDEA激活码,2021年最新IDEA永久激活码无偿分享!
    MQ消息队列(应用场景,产品对比)
  • 原文地址:https://www.cnblogs.com/dragen/p/3141898.html
Copyright © 2011-2022 走看看