zoukankan      html  css  js  c++  java
  • CYPEESS USB3.0程序解读之---GPIO

    CPRESS 官方给出的SDK1.1中(目前最新的SDK),提供了大量的例程供我们开发软件的时候作参考,就像STM32的开发一样提供了库一样,但是又不是库,仅仅是参考例程。

    首先看一个简单一点的GPIO的例子(GpioApp)

     

    1.先是一个错误处理的函数,我们不需要它,故这是一个死循环。

     

    2. CyFxDebugInit 这个函数,将串口作为调试口用115200bps。

     

    3.

    void CyFxGpioIntrCb (

    uint8_t gpioId /* Indicates thepin that triggered the interrupt */

    )

     这个函数是一个中断回调函数。必须在某个地方注册一下。

     它有下列过程:

    apiRetStatus=CyU3PGpioGetValue(gpioID,&gpioValue);

    这个函数得到某个端口中断的值

    这个gpioValue是一个BOOL值。而ID则是某一个端口的端口号。这个函数只能返回一个引脚。

    下面介绍这个ID是什么指定的。

    4.

    CyU3PEventSet(&glFxGpioAppEvent,CY_FX_GPIOAPP_GPIO_HIGH_EVENT,CYU3P_EVENT_OR); 

    如果为高,则设置一个事件。是一个高事件发生。注意到事件是一个全局变量,而这个事件中有许多参数,其中比较重要的是一个回调函数。应该在某个地方将这个事件与一个回调函数联系起来。

     

    5.voidCyFxGpioInit (void

    apiRetStatus =CyU3PGpioInit(&gpioClock,CyFxGpioIntrCB);

    这个函数是设定gpio的时钟,以及中断的回调函数。这与4中部分形成对照。

    然后将gpio45定义为输入且允许中断

        gpioConfig.intrMode = CY_U3P_GPIO_INTR_BOTH_EDGE;

        apiRetStatus = CyU3PGpioSetSimpleConfig(45, &gpioConfig);

    GPIO的21脚本来作为GPIF的控制信号的。不能用CyU3PDeviceConfigureIOMatrix来将它作为GPIF IOs.

    这个过载API调用必须进行必须小心当改变这个引脚的功能时。如果IO脚作为GPIF的一部分连到外部设备上。则它不能再作为GPIF IO使用。在这里CTL4是不使用的,所以用它用IO脚是安全的。

    apiRetStatus = CyU3PDeviceGpioOverride(21,CyTrue);

    接下来

    apiRetStatus =CyU3PGpioSetSimpleConfig(21,&gpioConfig);

    6. 接下来有两个线程,一个是输出线程,一个是输入线程,先看输出线程:

    apiRetStatus = CyFxDebugInit(); 

    初始化调试模式。这个在2中定义的。

    CyFxGpioInit(); 

    这个也在前面定义过。后面是一个闪灯程序。

    apiRetStatus = CyU3PGpioSetValue(21,true);

    将输出置为高。

    延时2秒,将输出变为低。

    延时2秒。

    7. 下面再来看输入线程:是一个循环,等事件发生。

    txApiRetStatus = CyU3PEventGet(&glFxGpioAppEvent,

    (CY_FX_GPIOAPP_GPIO_HIGH_EVENT |CY_FX_GPIOAPP_GPIO_LOW_EVENT),

    CYU3P_EVENT_OR_CLEAR,&eventFlag, CYU3P_WAIT_FOREVER);

    这里表示永远等下去。等到后要清除事件,另返回事件的标志,这个标志我们没有用。如果等到高的标志,就打印一个引脚为高,如果为低,就打印一个引脚为低的标志。估计这个等事件标志将被block.

    这样整个过程清楚了,IO脚触发引起一个中断。这个中断回调函数中将触发一个事件。在这个线程中将等事件发生,如果发生了,就打印出引脚的状态。

     

    事件在什么地方初始化呢?还是不需要初始化?

    8. 事件是要初始化的。在应用程序中初始化了,下面就看这个应用程序

    先创建一个输出线程。

    再创建一个输入线程

    然后 

    retThrdCreate =CyU3EventCreate(&glFxGpioAppEvent);

    9. 最后看一下main()

    main()中主要是将GPIO引脚初始化一下。

    io_cfg.gpioSimpleEn[0] = 0;

    io_cfg.gpioSimpleEn[1] = 0x00002000; /* GPIO 45 */

    io_cfg.gpioComplexEn[0] = 0;

    io_cfg.gpioComplexEn[1] = 0;

    45引脚为什么对应的是0x2000.?这是因为它是32位的,45引脚=32+13 ,这个第13位正好是0X2000(1<<13 就是0x2000)。

     

    *实例总结

    从main开始看起:

    再看一下几个定义:输出线程,输入线程及事件在文件一开始就定义了。

    CyU3PThread gpioOutputThread; /* GPIOthread structure */

    CyU3PThreadgpioInputThread; /* GPIO thread structure */

    CyU3PEventglFxGpioAppEvent; /* GPIO input event group. */

    它主要是调用了一个串口设置函数,然后就进入到cache控制设置,再后来就是设置一个IO脚,45脚使之使能。并且选用配置模式(即LPP模式)。允许了UART,不允许IIC,IIS,SPI,另外isDQ32bit也不允许。这个表示它不支持GPIF的32位模式。

     

    然后我们再看应用程序启动,这是由系统自动调用的。我们可能修改它的内容,但是它是必须的。

    这个函数中,它创建了两个线程。一个是输入线程,一个是输出线程。

    另外,容易遗忘的一件事是它创建了一个事件。事件的创建只要这样就可以了:     

    retThrdCreate = Cy3U3PEventCreate(&glFxGpioAppEvent);

    再往上,就是输入线程了。这个线程看输入引脚的变化,而这个变化由中断回调函数引起,中断回调函数中,它会产生一个事件,而我们的线程就监视这个事件。如果有事件高发生,就串口打印一个引脚高,如果低,就打印一个引脚低。看它是如何实现的:

    txApiRetStatus =CyU3PEventGet(&glFxGpioAppEvent,

    (CY_FX_GPIOAPPP_GPIO_HIGH_EVENT|CY_FX_GPIOAPP_GPIO_LOW_EVENT),

    CYU3P_EVENT_OR_CLEAR,&eventFlag,CYU3P_WAIT_FOREVER);

    这是个等事件的函数,这个函数无法找到它的定义,它是一个API函数。我们找API,发现它的参数含义。

    这里有一个CYU3P_EVENT_OR_CLEAR表示只要上面有一个位被设置就返回且清除标志。----OR。

    而真正的事件就放在标志中返回了。

    既然有读事件,就必有设置事件,事件的设置应该在中断回调中实现。而中断回调的注册,应该在初始化时实现。下面应该可以很快看到这点。---事实上,在下面的输出线程中就实现了注册。

     

    输出线程实现,输出线程比较有意思的是其DebugInit()居然是在它中间实现的,其实这个也可以在main中。

    而接下来,它又调用了初始化GpioInit()这个函数。在这个函数中,先初始化GPIO,这个GPIO居然还要将时钟也设置一下,有点不合常理。在这个初始化中,它还指明了GPIO中断回调函数的注册。尽管这个中断函数应该是在输入线程中注册似更合理一些。接下来,45脚要用之为输入,所以要将配置设一下:

    gpioConfig.outValue = CyTrue; //输出为高 因为是输入,要将它设为高

    gpioConfig.inputEn = CyTrue; //输入使能

    gpioConfig.driveLowEn = CyFalse; //不要驱动低也不要驱动高

    gpioConfig.driveHighEn = CyFalse;

    gpioConfig.intrMode = CY_U3P_GPIO_INTR_BOTH_EDGE; //允许中断

    apiRetStatus = CyU3PGpioSetSimpleConfig(45, &gpioConfig);


    如此这般配置了45脚。

    接下来,要配置21脚,因为21脚比较特殊本来是用于GPIF的CTRL4的。现在要使用它就要重载一下:

    这样的IO脚是不可以象在主程序中哪样,将它直接设为输出的,而是要先重载。

    同样,看输出脚是如何定义的

    gpioConfig.outValue = CyFalse; //低电平

    gpioConfig.driveLowEn = CyTrue; //允许低输出

    gpioConfig.driveHighEn = CyTrue; //允许高输出

    gpioConfig.inputEn = CyFalse; //方向设为输出

    gpioConfig.intrMode = CY_U3P_GPIO_NO_INTR; //不用中断

    再看一下回调函数,如何实现它的:

    当引脚有跳变时,这个函数被调用。首先,它得到引脚的值。这个回调函数是带参数的。当它发生时,会带过来一个参数。表明是哪一个引脚触发了这个事件。这在库函数中可能已经处理了,提供给用户程序就不用麻烦再去看原因了。我想可能有一个机制,即有一个中断状态寄存器,表示是哪一个引脚变化了。

    在这里调用了一个函数:

    CyU3PGpioGetValue(gpioId,&gpioValue); 


    注意到这个值是一个BOOL型的。

    然后根据情况来设置事件:

    CyU3PEventSet(&glFxGpioAppEvent,CY_FX_GPIOAPP_GPIO_HIGH_EVENT,CYU3P_EVENT_OR);

    我们看,其中有要设置的事件指针,有什么事件,以什么方式设置,它是以OR的方式设置的。这个OR表示的是将这个第2个参数与当前的事件标志进行或。显然,如果相或的话,则事件标志将被置1,而如果与则完全不同,它没效果。(在得到事件中,有一个AND表示全部标志都符合才生成事件,所以也是用OR的,不然,不可能全部符合的,永远不会发生事件了,因为不可能既变高又变低的)。

    至此整个程序解读完了。在这个例子中,没用到USB有关的部分。就是当一个MCU用的GPIO,因为这个芯片的CPU是ARM926EJ的内核,运行频率可以达到200MHz,和之前的FX2芯片的51内核完全不是一个级别,所以这个做主控芯片是完全可以的。
  • 相关阅读:
    IE内核浏览器下中文双引号自动显示成英文引号的说明
    【转】Linq初体验——Order By 通过属性名动态排序
    【转】Windows8不联网直接安装.Net 3.5 Framework的方法
    DrawImage在绘制图片的时候,为什么会擅自改变图片的大小?
    用js判断.net版本
    【转】html 页面打印并 分页
    利用IE9下载网络歌曲新法
    LINQ访问DataTable
    20165205与20165233结对感想以及创意照
    20165233 Java第一章学习总结
  • 原文地址:https://www.cnblogs.com/Zoran-/p/5819296.html
Copyright © 2011-2022 走看看