STM32MP157实现按键控制灯开关,蜂鸣器开关,风扇开关

1、key_it.c

#include"key_it.h"

//延时函数
void delay1(int ms)
{
	for(int i=0;i<ms;i++)
	{
		for(int j=0;j<2000;i++);
	}
}

//灯
void led1_init()
{
	//RCC使能
	RCC->MP_AHB4ENSETR |= (0x3<<4);
    //设置PE10为输出
    GPIOE->MODER &= (~(0X3<<20));
    GPIOE->MODER |= (0X1<<20);
    //设置推挽输出
    GPIOE->OTYPER &= (~(0X1<<10));
    //设置三个管脚低速输出
    GPIOE->OSPEEDR &= (~(0X3<<20));
    //设置三个管脚输出时无上拉电阻和下拉电阻    
    GPIOE->PUPDR &= (~(0X3<<20));
}

//蜂鸣器
void spi_init()
{
	//RCC使能
	RCC->MP_AHB4ENSETR |= (0x3<<1);
	//设置PB6为输出
	GPIOB->MODER &= (~(0x3<<12));
	GPIOB->MODER |= (0x1<<12);
	//设置推挽输出
	GPIOB->OTYPER &= (~(0x1<<6));
	//设置速度
	GPIOB->OSPEEDR &= (~(0x3<<12));
	//设置管脚输出时无上拉电阻和下拉电阻
	GPIOB->PUPDR &= (~(0x3<<12));
}

//风扇
void fan_init()
{	
	//RCC使能
	RCC->MP_AHB4ENSETR |= (0x3<<4);
    //设置PE10为输出
    GPIOE->MODER &= (~(0X3<<18));
    GPIOE->MODER |= (0X1<<18);
    //设置推挽输出
    GPIOE->OTYPER &= (~(0X1<<9));
    //设置三个管脚低速输出
    GPIOE->OSPEEDR &= (~(0X3<<18));
    //设置三个管脚输出时无上拉电阻和下拉电阻    
    GPIOE->PUPDR &= (~(0X3<<18));
	
}

//按键3的配置
void key3_it_config()
{
    //RCC使能GPIOF时钟
    RCC->MP_AHB4ENSETR |= (0x1<<5);
    GPIOF->MODER &= (~(0x3<<16));
    EXTI->EXTICR3 &=(~(0xFF<<0));
    EXTI->EXTICR3 |= (0x5<<0);
    EXTI->FTSR1 |= (0x1<<8);
    EXTI->C1IMR1 |= (0x1<<8);
    GICD->ISENABLER[3] |= (0X1<<2); 
    GICD->IPRIORITYR[24] &= (~(0X1F<<19)); 
    GICD->ITARGETSR[24]  &= (~(0X3<<16));
    GICD->ITARGETSR[24] |= (0X1<<16);
}

//按键1 按键2以及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));
    //pf7
    GPIOF->MODER &= (~(0x3<<14));
    //设置PF9 PF7 PF8产生EXTI事件 EXTI_EXTICRx
    //pf7
    EXTI->EXTICR2 &=(~(0xFF<<24));
    EXTI->EXTICR2 |= (0x5<<24);
    //pf8
    // EXTI->EXTICR3 &=(~(0xFF<<0));
    // EXTI->EXTICR3 |= (0x5<<0);
    //pf9
    EXTI->EXTICR3 &=(~(0xFF<<8));
    EXTI->EXTICR3 |= (0x5<<8);
    //设置事件触发方式为下降沿触发EXTI_FTSR1
    //pf7
    EXTI->FTSR1 |= (0x1<<7);
    //pf8
    // EXTI->FTSR1 |= (0x1<<8);
    //pf9
    EXTI->FTSR1 |= (0x1<<9);
    //设置EXTI事件触发不屏蔽EXTI_IMR1
    //PF7
    EXTI->C1IMR1 |= (0x1<<7);
    //PF8
    // EXTI->C1IMR1 |= (0x1<<8);
    //PF9
    EXTI->C1IMR1 |= (0x1<<9);
    //使能中断能转发到特定的CPU接口层GICD_ISENABLERx
    //PF7 97号中断
    GICD->ISENABLER[3] |= (0X1<<1); 
    //PF78 98号中断
    // GICD->ISENABLER[3] |= (0X1<<2); 
    //PF9 99号中断
    GICD->ISENABLER[3] |= (0X1<<3); 
    // GICD_ISENABLERx设置中断优先级
    //PF7
    GICD->IPRIORITYR[24] &= (~(0X1F<<11)); 
    //PF8
    // GICD->IPRIORITYR[24] &= (~(0X1F<<19)); 
    //PF9
    GICD->IPRIORITYR[24] &= (~(0X1F<<27)); 
    //设置当前中断被转发到哪一个CPU处理GICD_ITARGETSRx
    //pf7 97中断
    GICD->ITARGETSR[24]  &= (~(0X3<<8));
    GICD->ITARGETSR[24] |= (0X1<<8);
    //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;

    
}

do_irp.c

#include "key_it.h"
extern void printf(const char *fmt, ...);
unsigned int i = 0;
void do_irq(void) 
{
	led1_init();  //led初始化
	spi_init();  //蜂鸣器初始化
	fan_init();  //风扇初始化
	static int flag=0;
    int irqno;//保存中断号
    irqno=GICC->IAR & 0X3FF;
    switch(irqno)
    {
	case 99://key1
		//按键1中断处理

		if(0==flag)
		{
			GPIOE->ODR |= (0x1<<10);
			flag=1;
		}
		else if(1==flag)
		{
			GPIOE->ODR &= (~(0x1<<10));
			flag=0;
		}

		//清除挂起中断标志位GICD_ICPENDRx
		GICD->ICPENDR[3] |= (0x1<<3);
		//清除中断触发标志位EXTI_FPR1
		EXTI->FPR1 |= (0x1<<9);
		break;
	case 97://key2
		//按键2中断处理
		if(0==flag)
		{
			GPIOB->ODR |= (0x1<<6);
			flag=1;
		}
		else if(1==flag)
		{
			GPIOB->ODR &= (~(0x1<<6));
			flag=0;
		}

		//清除挂起中断标志位GICD_ICPENDRx
		GICD->ICPENDR[3] |= (0x1<<1);
		//清除中断触发标志位EXTI_FPR1
		EXTI->FPR1 |= (0x1<<7);
		break;
	case 98://key3
		//按键3中断处理
		
		if(0==flag)
		{
			GPIOE->ODR |= (0x1<<9);
			flag=1;
		}
		else if(1==flag)
		{
			GPIOE->ODR &= (~(0x1<<9));
			flag=0;
		}

		//清除挂起中断标志位GICD_ICPENDRx
		GICD->ICPENDR[3] |= (0x1<<2);
		//清除中断触发标志位EXTI_FPR1
		EXTI->FPR1 |= (0x1<<8);
		break;
    }
    //清除处理完的中断号GICC_EOIR
    GICC->EOIR =irqno;
}

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 key3_it_config();
void delay1(int ms);
void led1_init();
void spi_init();
void fan_init();

#endif

main.c

#include "key_it.h"
int main()
{
    char c;
    char *s;
    uart4_init();//串口初始化
    //中断初始化
    key_it_config();
	key3_it_config();
    while(1)
    {
        //保证主程序不结束
    }
    return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,可以通过以下步骤实现: 1. 配置两个按键引脚和蜂鸣器引脚的GPIO模式,其中按键引脚需要设置为输入模式,蜂鸣器引脚需要设置为输出模式。 2. 在主循环中读取按键引脚的输入状态,如果按键被按下,则设置一个变量标记按键被按下。 3. 在主循环中读取第二个按键引脚的输入状态,如果该按键被按下且之前的按键也被按下,则将蜂鸣器引脚的输出状态翻转一次,实现开关蜂鸣器的功能。 4. 在主循环中等待一段时间后再次读取按键引脚的输入状态,避免按键抖动造成误触发。 以下是一份示例代码,其中按键引脚为PA0和PA1,蜂鸣器引脚为PB0: ```c #include "stm32f4xx.h" int main(void) { // 使能GPIOA和GPIOB的时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA | RCC_AHB1Periph_GPIOB, ENABLE); // 配置PA0和PA1为输入模式 GPIO_InitTypeDef gpio_init; gpio_init.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1; gpio_init.GPIO_Mode = GPIO_Mode_IN; gpio_init.GPIO_Speed = GPIO_Speed_50MHz; gpio_init.GPIO_PuPd = GPIO_PuPd_UP; GPIO_Init(GPIOA, &gpio_init); // 配置PB0为输出模式 gpio_init.GPIO_Pin = GPIO_Pin_0; gpio_init.GPIO_Mode = GPIO_Mode_OUT; gpio_init.GPIO_Speed = GPIO_Speed_50MHz; gpio_init.GPIO_OType = GPIO_OType_PP; gpio_init.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_Init(GPIOB, &gpio_init); // 循环读取按键状态并控制蜂鸣器 int key1_pressed = 0; while (1) { // 读取按键1状态 int key1_state = GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0); // 如果按键1被按下,则标记按键1已经被按下 if (key1_state == 0) { key1_pressed = 1; } // 读取按键2状态 int key2_state = GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_1); // 如果按键2被按下并且按键1也被按下,则翻转蜂鸣器状态 if (key2_state == 0 && key1_pressed == 1) { GPIO_ToggleBits(GPIOB, GPIO_Pin_0); key1_pressed = 0; } // 延时一段时间 for (volatile int i = 0; i < 100000; i++); } } ``` 这份代码将会循环读取两个按键的状态,并在第二个按键被按下且之前的按键也被按下时翻转蜂鸣器的状态,实现了任意两个按键控制蜂鸣器开关的功能。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值