arm 按键控制led、风扇、蜂鸣器

 main.c

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

do_irq.c

#include "key_it.h"
extern void printf(const char *fmt, ...);
unsigned int i = 0;
void do_irq(void)
{
    int irqno; // 保存中断号
    printf("%d\n",irqno);
    irqno = GICC->IAR & 0X3FF;
    switch (irqno)
    {
    case 99: // key1
        // 按键1中断处理
		            if(GPIOE->ODR & (1<<10)){
                GPIOE->ODR &= (~(1<<10));
            }else{
                GPIOE->ODR |= (1<<10);
            }
			printf("KEY1 INT\n");
        // 清除挂起中断标志位GICD_ICPENDRx
        GICD->ICPENDR[3] |= (0x1 << 3);
        // 清除中断触发标志位EXTI_FPR1
        EXTI->FPR1 |= (0x1 << 9);
        break;
    case 97: // key2
        // 按键2中断处理

            if(GPIOB->ODR & (1<<6)){
                GPIOB->ODR &= (~(1<<6));
            }else{
                GPIOB->ODR |= (1<<6);
            }
        printf("KEY2 INT\n");
        // 清除挂起中断标志位GICD_ICPENDRx
        GICD->ICPENDR[3] |= (0x1 << 1);
        // 清除中断触发标志位EXTI_FPR1
        EXTI->FPR1 |= (0x1 << 7);
        break;
    case 98: // key3
        // 按键3中断处理
		if(GPIOE->ODR&(0x1<<9))
		{
			GPIOE->ODR&=(~(0x1<<9));
		}else
		{
			GPIOE->ODR|=(0x1<<9);
		}
        printf("KEY3 INT\n");
        // 清除挂起中断标志位GICD_ICPENDRx
        GICD->ICPENDR[3] |= (0x1 << 2);
        // 清除中断触发标志位EXTI_FPR1
        EXTI->FPR1 |= (0x1 << 8);
        break;
    }
    // 清除处理完的中断号GICC_EOIR
    GICC->EOIR = irqno;
}

key_it.c

#include"key_it.h"
void all_led_init()
{
    //RCC使能
    RCC->MP_AHB4ENSETR |= (0x3<<4);
    //设置PE10 PF10 PE8为输出
    GPIOE->MODER &= (~(0X3<<20));
    GPIOE->MODER |= (0X1<<20);
    GPIOF->MODER &= (~(0X3<<20));
    GPIOF->MODER |= (0X1<<20);
    GPIOE->MODER &= (~(0X3<<16));
    GPIOE->MODER |= (0X1<<16);
    //设置推挽输出
    GPIOE->OTYPER &= (~(0X1<<10));
    GPIOF->OTYPER &= (~(0X1<<10));
    GPIOE->OTYPER &= (~(0X1<<8));
    //设置三个管脚低速输出
    GPIOE->OSPEEDR &= (~(0X3<<20));
    GPIOF->OSPEEDR &= (~(0X3<<20));
    GPIOE->OSPEEDR &= (~(0X3<<16));
    //设置三个管脚输出时无上拉电阻和下拉电阻
    GPIOE->PUPDR &= (~(0X3<<20));
    GPIOF->PUPDR &= (~(0X3<<20));
    GPIOE->PUPDR &= (~(0X3<<16));

}
void fmq_init()
{
    //RCC使能
    RCC->MP_AHB4ENSETR  |= (0x1<<1);
    //设置PE9为输出
    GPIOB->MODER &= (~(0X3<<12));
    GPIOB->MODER |= (0X1<<12);
    //设置推挽输出
    GPIOB->OTYPER &= (~(0X1<<6));
    //设置三个管脚低速输出
    GPIOB->OSPEEDR &= (~(0X3<<12));
    //设置三个管脚输出时无上拉电阻和下拉电阻    
    GPIOB->PUPDR &= (~(0X3<<12));
}
void fs_init()
{
    //RCC使能
    RCC->MP_AHB4ENSETR  |= (0x1<<4);
    //设置PE9为输出
    GPIOE->MODER &= (~(0X3<<18));
    GPIOE->MODER |= (0X1<<18);
    //设置推挽输出
    GPIOE->OTYPER &= (~(0X1<<9));
    //设置三个管脚低速输出
    GPIOE->OSPEEDR &= (~(0X3<<18));
    //设置三个管脚输出时无上拉电阻和下拉电阻
    GPIOE->PUPDR &= (~(0X3<<18));
}
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;  
}

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 all_led_init();
void fs_init();
void fmq_init();
#endif

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是一个简单的C语言程序,演示了如何使用串口控制蜂鸣器: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <fcntl.h> #include <termios.h> #include <string.h> int main(int argc, char *argv[]) { int fd; struct termios oldtio, newtio; char buf[255] = {0}; // 打开串口 fd = open("/dev/ttySAC1", O_RDWR | O_NOCTTY); if (fd < 0) { printf("Open Serial Port Failed!\n"); return -1; } // 设置串口参数 tcgetattr(fd, &oldtio); memset(&newtio, 0, sizeof(newtio)); newtio.c_cflag = B115200 | CS8 | CLOCAL | CREAD; newtio.c_iflag = IGNPAR; newtio.c_oflag = 0; newtio.c_lflag = 0; newtio.c_cc[VTIME] = 0; newtio.c_cc[VMIN] = 1; tcflush(fd, TCIFLUSH); tcsetattr(fd, TCSANOW, &newtio); // 控制蜂鸣器 while (1) { memset(buf, 0, sizeof(buf)); read(fd, buf, sizeof(buf)); if (buf[0] == '1') { printf("Buzzer ON!\n"); // 控制蜂鸣器开启 // TODO: 在这里添加你的代码 } else { printf("Buzzer OFF!\n"); // 控制蜂鸣器关闭 // TODO: 在这里添加你的代码 } } // 关闭串口 close(fd); tcsetattr(fd, TCSANOW, &oldtio); return 0; } ``` 这个程序的主要思路是:首先打开串口 `/dev/ttySAC1`,然后设置串口通信参数,接着进入一个 while 循环,不断读取串口数据并判断,如果收到的数据是 `'1'`,则控制蜂鸣器开启,否则控制蜂鸣器关闭。需要注意的是,在程序中的 TODO 部分,你需要添加控制蜂鸣器的具体代码。 另外,你需要根据自己的实际情况修改串口设备文件名和波特率等参数。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值