ARM按键中断控制事件

设置按键中断,按键1按下,LED亮,再按一次,灭按键2按下,蜂鸣器响。再按一次,不响按键3按下,风扇转,再按一次,风扇停

src/key_it.c

#include"key_it.h"
//GPIO初始化
void all_led_init()
{
    //RCC使能
    RCC->MP_AHB4ENSETR  |= (0X1<<4);
    //设置PE10 PF10 PE8为输出
    GPIOE->MODER &= (~(0X3<<20));
    GPIOE->MODER |= (0X1<<20);
    //设置推挽输出
    GPIOE->OTYPER &= (~(0X1<<10)); 
    //设置三个管脚低速输出
    GPIOE->OSPEEDR &= (~(0X3<<20));
    //设置三个管脚输出时无上拉电阻和下拉电阻    
    GPIOE->PUPDR &= (~(0X3<<20));
}
void all_spi_init()
{
    //RCC使能
    RCC->MP_AHB4ENSETR  |= (0X3<<1);
    //设置PE10 PF10 PE8为输出
    GPIOB->MODER &= (~(0X3<<12));
    GPIOB->MODER |= (0X1<<12);
    //设置推挽输出
    GPIOB->OTYPER &= (~(0X1<<6)); 
    //设置三个管脚低速输出
    GPIOB->OSPEEDR &= (~(0X3<<12));
    //设置三个管脚输出时无上拉电阻和下拉电阻    
    GPIOB->PUPDR &= (~(0X3<<12));
}
void all_fan_init()
{
    //RCC使能
    RCC->MP_AHB4ENSETR  |= (0X3<<4);
    //设置PE10 PF10 PE8为输出
    GPIOE->MODER &= (~(0X3<<18));
    GPIOE->MODER |= (0X1<<18);
    //设置推挽输出
    GPIOE->OTYPER &= (~(0X1<<9)); 
    //设置三个管脚低速输出
    GPIOE->OSPEEDR &= (~(0X3<<18));
    //设置三个管脚输出时无上拉电阻和下拉电阻    
    GPIOE->PUPDR &= (~(0X3<<18));
}
//按键2的配置
void key2_it_config()
{
    //RCC使能GPIOF时钟
    RCC->MP_AHB4ENSETR |= (0x1<<5);
    //pf7
    GPIOF->MODER &= (~(0x3<<14));
    //pf7
    EXTI->EXTICR2 &=(~(0xFF<<24));
    EXTI->EXTICR2 |= (0x5<<24);
    //pf7
    EXTI->FTSR1 |= (0x1<<7);
    //PF7
    EXTI->C1IMR1 |= (0x1<<7);
    //PF7 97号中断
    GICD->ISENABLER[3] |= (0X1<<1); 
    //PF7
    GICD->IPRIORITYR[24] &= (~(0X1F<<11)); 
    //pf7 97中断
    GICD->ITARGETSR[24]  &= (~(0X3<<8));
    GICD->ITARGETSR[24] |= (0X1<<8);
}
//按键1 按键3以及GICC层的配置
void key_it_config()
{
    //RCC使能GPIOF时钟
    RCC->MP_AHB4ENSETR |= (0x1<<5);
    //设置PF9 PF7 PF8GPIO输入
    //PF9
    GPIOF->MODER &= (~(0x3<<18));
    //PF8
    GPIOF->MODER &= (~(0x3<<16));
    //设置PF9 PF7 PF8产生EXTI事件 EXTI_EXTICRx
    //pf8
    EXTI->EXTICR3 &=(~(0xFF<<0));
    EXTI->EXTICR3 |= (0x5<<0);
    //pf9
    EXTI->EXTICR3 &=(~(0xFF<<8));
    EXTI->EXTICR3 |= (0x5<<8);
    //设置事件触发方式为下降沿触发EXTI_FTSR1
    //pf8
    EXTI->FTSR1 |= (0x1<<8);
    //pf9
    EXTI->FTSR1 |= (0x1<<9);
    //设置EXTI事件触发不屏蔽EXTI_IMR1
    //PF8
    EXTI->C1IMR1 |= (0x1<<8);
    //PF9
    EXTI->C1IMR1 |= (0x1<<9);
    //使能中断能转发到特定的CPU接口层GICD_ISENABLERx
    //PF8 98号中断
    GICD->ISENABLER[3] |= (0X1<<2); 
    //PF9 99号中断
    GICD->ISENABLER[3] |= (0X1<<3); 
    //GICD_ISENABLERx设置中断优先级
    //PF8
    GICD->IPRIORITYR[24] &= (~(0X1F<<19)); 
    //PF9
    GICD->IPRIORITYR[24] &= (~(0X1F<<27)); 
    //设置当前中断被转发到哪一个CPU处理GICD_ITARGETSRx
    //pf8 98中断
    GICD->ITARGETSR[24]  &= (~(0X3<<16));
    GICD->ITARGETSR[24] |= (0X1<<16);
    //pf9 99中断
    GICD->ITARGETSR[24]  &= (~(0X3<<24));
    GICD->ITARGETSR[24] |= (0X1<<24);
    //使能组0转发中断GICD_CTLR
    GICD->CTRL |= 0X1;
    //设置中断优先级掩码GICC_PMR
    GICC->PMR |= (0x1F<<3);
 
    //使能CPU可以处理组0转发的中断GICC_CTLR
    GICC->CTRL |= 0X1;
 
    
}
     

src/do_irq.c

    #include "key_it.h"
    extern void printf(const char *fmt, ...);
    unsigned int i = 0;
    void do_irq(void) 
    {
        static int flag=0;
        static int flag1=0;
        static int flag2=0;
        int irqno;//保存中断号
            irqno=GICC->IAR & 0X3FF;
            switch(irqno)
            {
                case 99://key1
                    //按键1中断处理
                    //printf("KEY1 INT\n");
                        if(0==flag)
                        {
                            //LED1亮
                           GPIOE->ODR |= (0X1<<10);
                            flag=1;
                        }
                        else if(1==flag)
                        {
                            //LED1灭
                             GPIOE->ODR &= (~(0X1<<10));
                            flag=0;
                        }
            //清除挂起中断标志位GICD_ICPENDRx
                GICD->ICPENDR[3] |= (0x1<<3);
            //清除中断触发标志位EXTI_FPR1
                EXTI->FPR1 |= (0x1<<9);
                break;
            case 97://key2
                //按键2中断处理
                //printf("KEY2 INT\n");
                        if(0==flag1)
                        {
                            //蜂鸣器响
                           GPIOB->ODR |= (0X1<<6);
                            flag1=1;
                        }
                        else if(1==flag1)
                        {
                            //蜂鸣器不响
                             GPIOB->ODR &= (~(0X1<<6));
                            flag1=0;
                        }
        //清除挂起中断标志位GICD_ICPENDRx
        GICD->ICPENDR[3] |= (0x1<<1);
        //清除中断触发标志位EXTI_FPR1
        EXTI->FPR1 |= (0x1<<7);
                break;
            case 98://key3
                //按键3中断处理
               // printf("KEY3 INT\n");
                        if(0==flag2)
                        {
                            //风扇转
                           GPIOE->ODR |= (0X1<<9);
                            flag2=1;
                        }
                        else if(1==flag2)
                        {
                            //风扇停
                             GPIOE->ODR &= (~(0X1<<9));
                            flag2=0;
                        }
        //清除挂起中断标志位GICD_ICPENDRx
        GICD->ICPENDR[3] |= (0x1<<2);
        //清除中断触发标志位EXTI_FPR1
        EXTI->FPR1 |= (0x1<<8);
                break;
        }
        //清除处理完的中断号GICC_EOIR
        GICC->EOIR =irqno;
    }
 

include/key_it.h

#ifndef __KEY_IT_H__
#define __KEY_IT_H__
#include "stm32mp1xx_rcc.h"
#include "stm32mp1xx_gpio.h"
#include "stm32mp1xx_exti.h"
#include "stm32mp1xx_gic.h"
 
void key_it_config();
void key2_it_config();
void all_led_init();
void all_spi_init();
void all_fan_init();
 
#endif

main.c

#include "uart.h"
 
#include "key_it.h"
 
 
 
int main()
 
{
 
 
    uart4_init();//串口初始化
 
    all_led_init();
 
    //中断初始化
 
    key_it_config();
 
	key2_it_config();
 
    all_spi_init();
 
    all_fan_init();
 
    while(1)
 
    {
 
        //保证主程序不结束
 
    }  
 
	return 0;
 
 
}
 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32是一款基于ARM Cortex-M系列内核的32位微控制器,可用于开发嵌入式系统。按键中断和LED灯亮灭是嵌入式系统中非常常见的功能,下面我将用300字回答如何使用STM32的按键中断控制LED灯的亮灭。 首先,我们需要连接一个按键和一个LED灯到STM32微控制器的相应引脚上。假设按键连接到PA0引脚,LED灯连接到PC13引脚。 接下来,我们需要配置STM32的GPIO外设来控制这些引脚。首先,开启相关引脚的时钟。然后,将PA0引脚配置为输入模式,PC13引脚配置为输出模式。可以使用STM32的寄存器或者开发环境提供的库函数来完成这些配置。 然后,我们需要配置外部中断。配置PA0引脚所对应的外部中断线,使其可以检测到按键的状态变化。可以使用STM32的寄存器或者库函数来完成这个配置。 紧接着,我们编写中断处理函数。当按键状态发生变化,中断触发时,中断处理函数被调用。在该函数中,我们可以读取按键引脚的状态,如果按键被按下,我们将PC13引脚设置为高电平,LED灯亮起;如果按键被释放,我们将PC13引脚设置为低电平,LED灯熄灭。 最后,我们需要在主函数中启用中断。启用中断后,当按键状态发生变化时,中断处理函数将被调用。 以上就是使用STM32的按键中断控制LED灯亮灭的步骤。通过配置GPIO外设和外部中断,编写中断处理函数,我们能够实现按下按键时,LED灯亮起;释放按键时,LED灯熄灭的功能。这样就完成了按键中断控制LED灯亮灭的任务。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值