通过串口收发数据控制led亮灭

//led.h
#ifndef __LED_H__
#define __LED_H__
#define RCC1 (*(unsigned int *)0X50000A28)
#define GPIOE_MODER (*(unsigned int *)0X50006000)
#define GPIOF_MODER (*(unsigned int *)0X50007000)
#define GPIOE_OTYPER (*(unsigned int *)0X50006004)
#define GPIOE_OSPEEDR (*(unsigned int *)0X50006008)
#define GPIOE_OPUPDR (*(unsigned int *)0X5000600C)
#define GPIOE_ODR (*(unsigned int *)0X50006014)
#define GPIOF_OTYPER (*(unsigned int *)0X50007004)
#define GPIOF_OSPEEDR (*(unsigned int *)0X50007008)
#define GPIOF_OPUPDR (*(unsigned int *)0X5000700C)
#define GPIOF_ODR (*(unsigned int *)0X50007014)
void all_led_init();
void LED1_ON();
void LED2_ON();
void LED3_ON();
void LED1_OFF();
void LED2_OFF();
void LED3_OFF();


#endif 
//uart4.h

#ifndef __UART4_H__
#define __UART4_H__
#include "stm32mp1xx_uart.h"
#include"stm32mp1xx_rcc.h"
#include"stm32mp1xx_gpio.h"
#include"gpio.h"

void uart4_init();
void putchar(char a);
char getchar();
void puts(char *s);
void gets(char *s);
#endif
//led.c
#include "led.h"

void all_led_init()//led灯的初始化
{
    //GPIOE/GPIOF时钟使能
   RCC1  |= (0x3<<4);
    //PE10设置为输出
    GPIOE_MODER &= (~(0X3<<20));
    GPIOE_MODER |= (0X1<<20);
    //PF10设置为输出
    GPIOF_MODER &= (~(0X3<<20));
    GPIOF_MODER |= (0X1<<20);
    //PE8设置为输出
    GPIOE_MODER &= (~(0X3<<16));
    GPIOE_MODER |= (0X1<<16);
    //PE10设置为推挽输出
    GPIOE_OTYPER &= (~(0X1<<10));
    //PF10设置为推挽输出
    GPIOF_OTYPER &= (~(0X1<<10));
    //PE8设置为推挽输出
    GPIOE_OTYPER &= (~(0X1<<8));
    //PE10设置为低速输出
    GPIOE_OSPEEDR &= (~(0X3<<20));
    //PF10设置为低速输出
    GPIOF_OSPEEDR &= (~(0X3<<20));
    //PE8设置为低速输出
   GPIOE_OSPEEDR &= (~(0X3<<16));
    //PE10无上拉下拉电阻
    GPIOE_OPUPDR &= (~(0X3<<20));
    //PF10无上拉下拉电阻
    GPIOF_OPUPDR &= (~(0X3<<20));
    //PE8无上拉下拉电阻
    GPIOE_OPUPDR &= (~(0X3<<16));

    //三盏灯默认输出低电平
    GPIOE_ODR &= (~(0X1<<10));
   GPIOF_ODR &= (~(0X1<<10));
    GPIOE_ODR &= (~(0X1<<8));
}

//led1亮
void LED1_ON()
{
    GPIOE_ODR |= (0X1<<10);
}
//led2亮
void LED2_ON()
{
   GPIOF_ODR |= (0X1<<10);
}
//led1亮
void LED3_ON()
{
    GPIOE_ODR |= (0X1<<8);
}

//LED1灭
void LED1_OFF()
{
     GPIOE_ODR &= (~(0X1<<10));
}
//LED2灭
void LED2_OFF()
{
     GPIOF_ODR &= (~(0X1<<10));
}
//LED3灭
void LED3_OFF()
{
     GPIOE_ODR &= (~(0X1<<8));
}
//uart4.c

#include "uart4.h"
#include "gpio.h"


void uart4_init()
{
    //使能GPIOB GPIOG UART4外设时钟
    RCC->MP_AHB4ENSETR |= (0x1<<1);//GPIOB
    RCC->MP_AHB4ENSETR |= (0x1<<6);//GPIOG
    RCC->MP_APB1ENSETR |= (0X1<<16);//UART4
    //设置PB2和PG11管脚复用
    //PB2
    GPIOB->MODER &= (~(0x3<<4));
    GPIOB->MODER |=(0x2<<4); 
    GPIOB->AFRL &= (~(0XF<<8));
    GPIOB->AFRL |= (0x8<<8);
    //PG11
    GPIOG->MODER &= (~(0x3<<22));
    GPIOG->MODER |=(0x2<<22); 
    GPIOB->AFRH &= (~(0XF<<12));
    GPIOB->AFRH |= (0x6<<12);
    //设置串口不使能 UE=0
    USART4->CR1 &= (~(0X1));
    //设置8位数据位
    USART4->CR1 &= (~(0x1<<12));
    USART4->CR1 &= (~(0x1<<28));
    //设置没有奇偶校验位
    USART4->CR1 &= (~(0x1<<10));
    //设置1位停止位
    USART4->CR2 &= (~(0x3<<12));
    //设置16倍过采样
    USART4->CR1 &= (~(0x1<<15));
    //设置时钟不分频
    USART4->PRESC &= (~(0xF));
    //设置波特率为115200
    USART4->BRR=0X22B;
    //使能发送器
    USART4->CR1 |= (0x1<<3);
    //使能接收器
     USART4->CR1 |= (0x1<<2);
    //使能uart4
     USART4->CR1 |= (0x1<<0);
    
}

//发送一个字符
void putchar(char a)
{
    //先判断发送数据寄存器是否为空
     //不为空阻塞等待
    while(!(USART4->ISR&(0x1<<7)));
    //为空向发送数据寄存器写入a的数据
   USART4->TDR=a;//写入数据
    //写入完成需要判断发送是否完成,不完成阻塞等待,完成了则函数结束
    while(!(USART4->ISR&(0x1<<6)));
}

//接收一个字符
char getchar()
{
    char a;
    //先判断接收数据寄存器中有没有准备好的数据
    //如果数据没有准备好则阻塞等待
    while(!(USART4->ISR&(0x1<<5)));
    //如果数据准备好了则读取
    a=USART4->RDR;
    //读取完毕将读取到的数据返回
    return a;
}

//发送字符串
void puts(char *s)
{
    while(*s)
    {
        putchar(*s);
        s++;
    }
    putchar('\n');// '\n'切换到下一行
    putchar('\r');//'\n'回车键,切换到行头

}

//接收字符串函数
void gets(char *s)
{
    while(1)
    {
        *s=getchar();
        putchar(*s);
        if((*s)=='\r')
            break;
        s++;
    }
    *s='\0';
    putchar('\n');//换行
}
#include"uart4.h"

#include "led.h"

//封装延时函数

void delay(int ms)

{

    int i,j;

    for(i=0;i<ms;i++)

    {

        for(j=0;j<2000;j++)

        {}



    }

}

int main()

{

    //串口初始化

    uart4_init();

    //led灯初始化

    all_led_init();

    char i;

    while(1)

    {

        i=getchar();

        if(i=='A')

        {

            LED1_ON();

        }

        else if(i=='a')

        {

            LED1_OFF();

        }

        else if(i=='B')

        {

            LED2_ON();

        }

        else if(i=='b')

        {

            LED2_OFF();

        }

        else if(i=='C')

        {

            LED3_ON();

        }

        else if(i=='c')

        {

            LED3_OFF();

        }

        else

        {

            puts("please enter abc");

        }

        

    }

    return 0;

}

好的,针对您的问题,我可以提供以下思路: 1. 在主函数中,开启串口接收,并且配置串口接收的回调函数。 2. 在串口接收回调函数中,根据接收到的数据进行LED控制。 以下是一个简单的代码示例,仅供参考: ```c #include "stm32f10x.h" #define LED_GPIO_PORT GPIOB #define LED_GPIO_PIN GPIO_Pin_12 void LED_Control(uint8_t state) { if(state == 0) { GPIO_ResetBits(LED_GPIO_PORT, LED_GPIO_PIN); } else if(state == 1) { GPIO_SetBits(LED_GPIO_PORT, LED_GPIO_PIN); } } void USART1_IRQHandler(void) { if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { uint8_t data = USART_ReceiveData(USART1); // 接收到数据,调用LED控制函数 LED_Control(data); } } int main(void) { GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; // 使能GPIOB和USART1时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); // 配置GPIOB.12引脚为推挽输出 GPIO_InitStructure.GPIO_Pin = LED_GPIO_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(LED_GPIO_PORT, &GPIO_InitStructure); // 配置USART1 USART_InitStructure.USART_BaudRate = 9600; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART1, &USART_InitStructure); // 配置串口接收中断优先级 NVIC_InitTypeDef NVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); // 开启串口接收中断 USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); // 启动USART1 USART_Cmd(USART1, ENABLE); while(1) { // 主函数中可以做一些其他的事情 } } ``` 需要注意的是,上述代码仅提供了一个简单的思路,具体的实现方式可能因为项目需求和硬件环境等原因而有所不同。如果您需要更加详细和专业的帮助,建议您咨询相关的技术人员或者查阅相关的资料。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值