zoukankan      html  css  js  c++  java
  • PIC32MZ tutorial -- Change Notification

      In my last post I implement "Key Debounce" with port polling, port polling is not very efficient. And this time, I will use change notification instead of port polling. It generates interrupt and starts debounce when the level of digital port changes, so it will eliminate the MCU load of port polling. 

      On my PIC32MZ EC Starter Kit, every I/O port pin (RAx-RKx) can be used as a change notification pin (CNAx-CNKx). The CN pins are configured as follows:

      1.Disable CPU interrupts.
      2.Set the desired CN I/O pin as an input by setting the corresponding TRISx register bits = 1.
      3.Enable the CN Module by setting the ON bit (CNCONx<15>) = 1.
      4.Enable individual CN input pins, enable optional pull-ups or pull-downs.
      5.Read the corresponding PORTx registers to clear the CN interrupt.
      6.Configure the CN Interrupt Priority bits, CNIP<2:0> (IPC6<20:18>), and Sub-priority bits,CNIS<1:0> (IPC6<17:16>).
      7.Clear the CN Interrupt Flag bit, by setting the CNIF bit (IFS1<0>) = 0.
      8.Configure the CN pin interrupt for a specific edge detect using the EDGEDETECT bit in the CNCONx register, and set up edge control using the CNENx/CNNEx bits.
      9.Enable the CN Interrupt Enable bit, by setting the CNIE bit (IEC1<0>) = 1.
    10.Enable CPU interrupts.

      The CNSTATx registers indicate whether a change occurred on the corresponding pin since the last read of the PORTx bit. It is ridiculous that data sheet or reference manual mentions CNSTATx register but gives no figure or details. After so many times retry, I get to know it has to clear CNSTATx bit corresponding to CN pin in the CN interrupt service routine. Otherwise, the interrupt service routine may not work. 

      Anyway, I get this change notification application work, and the following is the implementation.

      TIMER module:

    void Timer1_Init(void)
    {
        T1CON = 0x8010;
        PR1 = 0x30D3;
        IPC1SET = 0x5;
        TMR1 = 0;
        IEC0SET = 0x10;
        IFS0CLR = 0x10;
    }
    
    void Timer1_Write(unsigned int value)
    {
        TMR1 = value & 0xFFFF;
    }

      KEY module ( enable change notification interrupt, debounce function):

    #define DEBOUNCE_Input (PORTB & 0x1000)
    #define DEBOUNCE_Open() ANSELB = 0xFFFFEFFF
    #define DEBOUNCE_IOCtl() CNPUBSET = 0x1000
    #define DEBOUNCE_Output LedState
    #define DEBOUNCE_ThreholdLow 0
    #define DEBOUNCE_ThreholdHigh 100
    //extern volatile unsigned char DEBOUNCE_TimeFlag;
    extern volatile unsigned char DEBOUNCE_EventStart;


      volatile unsigned char DEBOUNCE_EventStart; //volatile unsigned char DEBOUNCE_TimeFlag; unsigned long DEBOUNCE_PreInput; unsigned int DEBOUNCE_Integrator; void Key_Init(void) { DEBOUNCE_EventStart = 0; DEBOUNCE_Integrator = DEBOUNCE_ThreholdHigh / 2; //DEBOUNCE_TimeFlag = 0; DEBOUNCE_Open(); DEBOUNCE_IOCtl(); DEBOUNCE_PreInput = DEBOUNCE_Input; CNENBSET = 0x1000; IPC29SET = 0x12000000; IFS3CLR = 0x800000; IEC3SET = 0x800000; CNCONBSET = 0x8000; } void Key_Debounce(void) { if (DEBOUNCE_EventStart) { if (DEBOUNCE_Input == 0) { if (DEBOUNCE_Integrator-- == DEBOUNCE_ThreholdLow) { DEBOUNCE_Output = 0; DEBOUNCE_PreInput = DEBOUNCE_Input; DEBOUNCE_EventStart = 0; DEBOUNCE_Integrator = DEBOUNCE_ThreholdHigh / 2; } } else //if (DEBOUNCE_Input == 1) { if (DEBOUNCE_Integrator++ == DEBOUNCE_ThreholdHigh) { DEBOUNCE_Output = 1; DEBOUNCE_PreInput = DEBOUNCE_Input; DEBOUNCE_EventStart = 0; DEBOUNCE_Integrator = DEBOUNCE_ThreholdHigh / 2; } } } }

      LED module:

    #define LED_IOCTL()       TRISHCLR = (1<<0)
    #define LED_SETON()       LATHSET = (1<<0)
    #define LED_SETOFF()      LATHCLR = (1<<0)
    #define LED_ONOFF()       LATHINV = (1<<0)
    #define LED_OPEN()        ANSELH &= 0xFFFFFFFE
    typedef enum _LED_STATE_t {
            /* Describe structure member. */
            OFF = 0,
    
            /* Describe structure member. */
            ON = 1
    
        } LED_STATE_t;
    LED_STATE_t PreLedState, LedState;
    
    void Led_Init(void)
    {
        LED_OPEN();
        LED_IOCTL();
        LED_SETON();
        LedState = ON;
        PreLedState = LedState;
    }
    
    void Led_Scheduler(void)
    {
        if (LedState != PreLedState)
        {
            LED_ONOFF();
            PreLedState = LedState;
        }
    }

      MAIN module (main function, timer1 ISR, CNB ISR):

    #include <xc.h>
    #include "Led.h"
    #include "Key.h"
    #include "Timer.h"
    #include "Interrupt.h"
    #include <sys/attribs.h>
    #include "ConfigurationBits.h"
    
    void __ISR(_CHANGE_NOTICE_B_VECTOR,ipl4AUTO) CNB_Handler(void)
    {
        if ((CNSTATB & 0x1000) == 0x1000)
        {
            DEBOUNCE_EventStart = 1;
            CNSTATBCLR = 0x1000;
            unsigned long tmp = DEBOUNCE_Input;
        }
        IFS3CLR = 0x800000;
    }
    
    void __ISR(_TIMER_1_VECTOR,ipl1AUTO) Timer1_Handler(void)
    {
        //DEBOUNCE_TimeFlag = 1;
        Key_Debounce();
        Timer1_Write(0);
        IFS0CLR = 0x10; // Clear flag
     }
    
    void main(void) 
    {
        Led_Init();
        Key_Init();
        Timer1_Init();
        Mvec_Interrupt();
        
        while (1)
        {
            Led_Scheduler();
        }
    }

       The application runs well on PIC32MZ EC Starter Kit, SW1(connects to RB12) pressed, LED1(RH0) is off, SW1 released, LED1 is on.

  • 相关阅读:
    STL中的distance和advance的简单用法
    Excel 根据数据 快捷生成sql语句
    vi | vim 用法
    常用windows 命令
    .NETCore3.0 + EFCore中使用Oracle报“ORA-12154: TNS:could not resolve the connect identifier specified"的错误处理
    CentOS虚拟机上安装java
    Eclipse快捷键
    Spring学习笔记
    ifconfig: command not found(CentOS专版,其他的可以参考)
    利用正则表达式截取带有嵌套方括号中最内层的字符串, 无论嵌套多少层始终要最里面的方括号的内容
  • 原文地址:https://www.cnblogs.com/geekygeek/p/pic32mz_changenotice.html
Copyright © 2011-2022 走看看