如何退出中断程序并返回主程序?

时间:2018-10-25 04:02:20

标签: c

我正在将Tiva C系列板与Keil uVision IDE一起使用。我基本上是在尝试使用中断执行以下操作:

如果未按下开关1和开关2:红色LED闪烁。 如果按下开关1,则进入中断程序并闪烁绿色LED。

释放开关1后,请重新输入电源以继续闪烁红色LED。

在下降沿触发时进入中断服务程序。我知道我需要以某种方式回到上升缘触发上,但是我不确定如何做到这一点。

我尽了最大的努力来评论,所以这是我到目前为止的内容:

#include <stdint.h>
#include <stdbool.h>
#include "Final Project.h"
#include "inc/hw_types.h"
#include "inc/hw_memmap.h"
#include "inc/hw_gpio.h"
#include "driverlib/sysctl.h"
#include "driverlib/pin_map.h"
#include "driverlib/gpio.h"

#include "inc/tm4c123gh6pm.h"       // manually added by the programmer
#include "driverlib/interrupt.h"    // manually added by the programmer


// Interrupt handler
void GPIOPortF_Handler(void)
    {
        // acknowledge flag for PF0
        GPIO_PORTF_ICR_R |= 0x01;   

        // Switch 1 is pressed
        if(GPIOPinRead(GPIO_PORTF_BASE, GPIO_PIN_4)==0x00)
        {
            // turn off red LED
            GPIO_PORTF_DATA_R &= ~0x02;
            // turn off blue LED
            GPIO_PORTF_DATA_R &= ~0x04;

            // while switch 1 is pressed
            while(GPIOPinRead(GPIO_PORTF_BASE, GPIO_PIN_4)==0x00)
            {
                // Turn on green LED
                GPIO_PORTF_DATA_R |= 0x08;
                // Delay
                SysCtlDelay(16000000/3/2);
                // Turn off green LED
                GPIO_PORTF_DATA_R &= ~0x08;
                // Delay
                SysCtlDelay(16000000/3/2);
            }
        }
    }

    void Interrupt_Init(void)
{
    NVIC_EN0_R |= 0x40000000;           // enable interrupt 30 in NVIC (GPIOF)
    NVIC_PRI7_R &= ~0x00E00000;     // configure GPIOF interru  
    GPIO_PORTF_IM_R |= 0x01;            // arm interrupt on PF0
    GPIO_PORTF_IS_R &= ~0x01;           // PF0 is edge-sensitive
    GPIO_PORTF_IBE_R |= 0x01;           // PF0 both edges trigger
    //GPIO_PORTF_IEV_R &= ~0x01;    // PF0 falling edge event
    //IntGlobalEnable();                    // Globally enable interrupt (without PinMux)
    IntMasterEnable();                      // Globally enable interrupt (with PinMux)
}


void
PortFunctionInit(void)
{
    // Enable Peripheral Clocks 
    SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOD);
    SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOF);

    // Enable pin PD1 for GPIOInput
    GPIOPinTypeGPIOInput(GPIO_PORTD_BASE, GPIO_PIN_1);

    // Enable pin PD0 for GPIOOutput
    GPIOPinTypeGPIOOutput(GPIO_PORTD_BASE, GPIO_PIN_0);

    // Enable pin PF2 for GPIOOutput
    GPIOPinTypeGPIOOutput(GPIO_PORTF_BASE, GPIO_PIN_2);

    // Enable pin PF3 for GPIOOutput
    GPIOPinTypeGPIOOutput(GPIO_PORTF_BASE, GPIO_PIN_3);

    // Enable pin PF0 for GPIOInput
        GPIO_PORTF_PUR_R |= 0x01; 

    // Enable pin PF4 for GPIOInput
        GPIO_PORTF_PUR_R |= 0x10; 

    //First open the lock and select the bits we want to modify in the GPIO commit register.
    HWREG(GPIO_PORTF_BASE + GPIO_O_LOCK) = GPIO_LOCK_KEY;
    HWREG(GPIO_PORTF_BASE + GPIO_O_CR) = 0x1;

    //Now modify the configuration of the pins that we unlocked.
    GPIOPinTypeGPIOInput(GPIO_PORTF_BASE, GPIO_PIN_0);

    // Enable pin PF1 for GPIOOutput
    GPIOPinTypeGPIOOutput(GPIO_PORTF_BASE, GPIO_PIN_1);

    // Enable pin PF4 for GPIOInput
    GPIOPinTypeGPIOInput(GPIO_PORTF_BASE, GPIO_PIN_4);
}


int main(void)
{

        // Iinitialize the GPIO ports   
        PortFunctionInit();
        // Configure the GPIOF interrupt
        Interrupt_Init();


    // Loop forever.
    while(1)
    {
            // Turn on red lED
            GPIO_PORTF_DATA_R |= 0x02;
            // Delay
            SysCtlDelay(16000000/3/2);
            // Turn off red lED
            GPIO_PORTF_DATA_R &= ~0x02;
            // Delay
            SysCtlDelay(16000000/3/2);
    }

}

1 个答案:

答案 0 :(得分:2)

  1. 在中断处理程序GPIOPortF_Handler中具有延迟功能是一个非常糟糕的主意。

  2. 此外,中断处理程序中还会有一个while循环,等待用户输入。

上述问题是中断将在一段时间内不存在。一个好的设计可以确保Interrupt上下文仅在短时间内运行并退出。这样可以注册其他中断。

我建议您在中断处理程序中设置一个易失性标志,并检查main中的标志。根据此标志,您可以采取适当的措施。

下面是一个简单的例子。

void GPIOPortF_Handler(void)
{
    GPIO_PORTF_ICR_R |= 0x01;   
    flags = 1;
} 

在主功能中

int main(void) 
{
    while(1) 
    {   
        if(flags==1)
        { 
            GPIO_PORTF_DATA_R |= 0x02; 
            SysCtlDelay(16000000/4); 
            GPIO_PORTF_DATA_R &= ~0x02; 
            SysCtlDelay(16000000/4); 
        } 
        else
        { 
            // Turn on red lED
            GPIO_PORTF_DATA_R |= 0x02;
            // Delay
            SysCtlDelay(16000000/4);
            // Turn off red lED
            GPIO_PORTF_DATA_R &= ~0x02;
            // Delay
            SysCtlDelay(16000000/4);
        }

        /* Check status of switch */
        if ( /* enter switch port here */ == 0)
        {
            GPIO_PORTF_DATA_R &= ~0x02; /* make green LED OFF */
            flags = 0;
        }
    }
}

请注意,此示例确实有一些缺点,并且将需要一些时间来检测密钥。根据时间SysCtlDelay,这是可以接受的。