结构体封装自定义命令启动led

gpio.h:

#ifndef __GPIO_H__
#define __GPIO_H__

#include "stm32mp1xx_gpio.h"
#include "stm32mp1xx_rcc.h"

#define RCC_AHB4_ENSETR (*(volatile unsigned int*)0x50000A28)

//寄存器模式初始化
typedef enum
{
    Input,
    Output,
    Alt_fun,
    Analog,
}gpio_moder_t;
//寄存器输出类型
typedef enum
{
    Push_pull,
    Open_drain,
}gpio_otyper_t;
//寄存器输出速度
typedef enum
{
    Low,
    Medium,
    High,
    Very_high,
}gpio_ospeedr_t;
//寄存器是否需要上下拉
typedef enum
{
    NO_PUPDR,
    Pull_up,
    Pull_down,
    Reserved,
}gpio_pupdr_t;

//对GPIO初始化
typedef struct
{
    gpio_moder_t moder;
    gpio_otyper_t otyper;
    gpio_ospeedr_t ospeedr;
    gpio_pupdr_t pupdr;
}gpio_init_t;

//对gpio引脚状态封装
typedef enum
{
    RESET,
    SET,
}gpio_status_t;
//引脚封装
#define GPIO_PIN_0     0
#define GPIO_PIN_1     1
#define GPIO_PIN_2     2
#define GPIO_PIN_3     3
#define GPIO_PIN_4     4
#define GPIO_PIN_5     5
#define GPIO_PIN_6     6
#define GPIO_PIN_7     7
#define GPIO_PIN_8     8
#define GPIO_PIN_9     9
#define GPIO_PIN_10 10
#define GPIO_PIN_11 11
#define GPIO_PIN_12 12
#define GPIO_PIN_13 13
#define GPIO_PIN_14 14
#define GPIO_PIN_15 15

void hal_gpio_init(gpio_t* gpiox,gpio_init_t* init,unsigned int pin);
void hal_gpio_write(gpio_t* gpiox,unsigned int pin,gpio_status_t state);
char my_strcmp(const char *src,const char *dth);

//void hal_gpio_output_high(unsigned int LEDx);
//void hal_gpio_output_low(unsigned int LEDx);

//结构体封装
typedef struct
{
    char* cmd_arr;
    gpio_t* gpiox;
    unsigned int pin;
    gpio_status_t status;
    void(*gpio_write_pin)(gpio_t* gpiox,unsigned int pin,gpio_status_t status);
}cmd_t;



#endif

gpio.c:

#include "../include/gpio.h"

void hal_gpio_init(gpio_t* gpiox,gpio_init_t* init,unsigned int pin)
{
    //1.设置GPIO引脚模式
    gpiox->MODER &=(~(0x3<<pin*2));
    gpiox->MODER |=(init->moder<<pin*2);
    //2.设置GPIO引脚输出模式
    gpiox->OTYPER &=(~(0x1<<pin));
    gpiox->OTYPER |=(init->otyper<<pin);
    //3.设置GPIO引脚输出速率
    gpiox->OSPEEDR &=(~(0x3<<pin*2));
    gpiox->OSPEEDR |=(init->ospeedr<<pin*2);
    //4.设置GPIO引脚是否需要上下拉
    gpiox->PUPDR &=(~(0x3<<pin*2));
    gpiox->PUPDR |=(init->pupdr<<pin*2);
}
void hal_gpio_write(gpio_t* gpiox,unsigned int pin,gpio_status_t state)
{
    //点亮状态
    if(state==SET)
    {
        gpiox->ODR |=(0x1<<pin);
    }
    else
        gpiox->ODR &=(~(0x1<<pin));//熄灭状态
}

char my_strcmp(const char *src,const char *dth)
{
    while(*dth!='\0'&&*src!='\0')
    {
        if(*src==*dth)
        {
            src++;
            dth++;
        }
        else
            return 0;
    }
    return 1;
}

uart.h:

#ifndef __UART4_H__
#define __UART4_H__

#include "stm32mp1xx_uart.h"
#include "stm32mp1xx_gpio.h"
#include "stm32mp1xx_rcc.h"

//串口初始化
void hal_uart_init();

//发送字符
void uart_put_char(const char ch);

//接收字符
char uart_get_char();

//发送字符串
void uart_put_string(const char* str);

//接收字符串
char* uart_get_string();

#endif

uart.c:

#include "../include/uart4.h"
extern void delay_ms(int ms);
//串口初始化
void hal_uart_init()
{
    //UART4_TX-->PG11  UART4_RX-->PB2
    /********RCC章节初始化**********/
    //使能GPIO组控制器
    RCC->MP_AHB4ENSETR |= (0x1<<1);
    RCC->MP_AHB4ENSETR |= (0x1<<6);
    //使能UART4控制器
    RCC->MP_APB1ENSETR |= (0x1<<16);

    /*********GPIO章节初始化**********/
    GPIOB->MODER &= (~(0x3<<4));
    GPIOB->MODER |= (0x2<<4);//将PB2模式设置为复用功能
    GPIOG->MODER &= (~(0x3<<22));
    GPIOG->MODER |= (0x2<<22);//将PG11模式设置为复用功能
    //设置引脚功能复用模式
    GPIOB->AFRL &= (~(0xf<<8));
    GPIOB->AFRL |= (0x8<<8);//UART4_TX[11:8]=0x1000
    GPIOG->AFRH &= (~(0xf<<12));
    GPIOG->AFRH |= (0x6<<12);//UART4_TX[15:12]=0x0110

    /***********UART章节初始化*************/
    if(USART4->CR1 & 0x1)
    {
        delay_ms(500);
        USART4->CR1 &= (~(0x1));
    }
    //设置位宽为8位
    USART4->CR1 &= (~(0x1<<28));
    USART4->CR1 &= (~(0x1<<12));
    //设置16倍采样率
    USART4->CR1 &= (~(0x1<<15));
    //设置无奇偶校验位
    USART4->CR1 &= (~(0x1<<10));
    //设置串口不分频
    USART4->PRESC &= (~(0xf));
    //设置1位停止位
    USART4->CR2 &= (~(0x3<<12));
    //设置波特率
    //USART4->BRR &=(~(0xffff));
    USART4->BRR |=0x22b;
    //设置发送寄存器使能
    USART4->CR1 |= (0x1<<3);
    //设置接收寄存器使能
    USART4->CR1 |= (0x1<<2);
    //设置串口使能
    USART4->CR1 |= (0x1);
}

//发送字符
void uart_put_char(const char ch)
{
    //循环判断发送寄存器是否为空
    while(!(USART4->ISR & (0x1<<7)));
    //为空则将字符放到发送寄存器中
    if(ch=='\n')
        uart_put_char('\r');
    USART4->TDR = ch;
    //循环判断是否发送完成
    while(!(USART4->ISR & (0x1<<6)));
}

//接收字符
char uart_get_char()
{
    //循环判断接收寄存器中是否有数据
    while(!(USART4->ISR & (0x1<<5)));
    char ch;
    ch = USART4->RDR;
    return ch;
}

//发送字符串
void uart_put_string(const char* str)
{
    while(*str)
    {
        uart_put_char(*str);
        str++;
    }
}

char buf[128]="";
//接收字符串
char* uart_get_string()
{
    char c;
    int i;
    for (i = 0; i < 126; i++)
    {
        c=uart_get_char();
        if(c=='\r')
            break;
        buf[i]=c;
        uart_put_char(c);
    }
    buf[i+1]='\0';
    uart_put_char('\n');
    return buf;
}

main.c:

#include "./include/uart4.h"
#include "./include/gpio.h"
extern void printf(const char *fmt, ...);
void delay_ms(int ms)
{
    int i,j;
    for(i = 0; i < ms;i++)
        for (j = 0; j < 1800; j++);
}
cmd_t cmd_brr[6]={
        [0]={
            .cmd_arr="led1on",
            .gpiox=GPIOE,
            .pin=GPIO_PIN_10,
            .status=SET,
            .gpio_write_pin=hal_gpio_write,
        },
        [1]={
            .cmd_arr="led1off",
            .gpiox=GPIOE,
            .pin=GPIO_PIN_10,
            .status=RESET,
            .gpio_write_pin=hal_gpio_write,
        },
        [2]={
            .cmd_arr="led2on",
            .gpiox=GPIOF,
            .pin=GPIO_PIN_10,
            .status=SET,
            .gpio_write_pin=hal_gpio_write,
        },
        [3]={
            .cmd_arr="led2off",
            .gpiox=GPIOF,
            .pin=GPIO_PIN_10,
            .status=RESET,
            .gpio_write_pin=hal_gpio_write,
        },
        [4]={
            .cmd_arr="led3on",
            .gpiox=GPIOE,
            .pin=GPIO_PIN_8,
            .status=SET,
            .gpio_write_pin=hal_gpio_write,
        },
        [5]={
            .cmd_arr="led3off",
            .gpiox=GPIOE,
            .pin=GPIO_PIN_8,
            .status=RESET,
            .gpio_write_pin=hal_gpio_write,
        }
    };
void hal_init()
{
    //RCC初始化
    RCC_AHB4_ENSETR |=(0x3<<4);
    gpio_init_t init={Output,Push_pull,Low,NO_PUPDR};
    //串口初始化
    void hal_uart_init();
    //LED初始化
    hal_gpio_init(GPIOE,&init,GPIO_PIN_10);
    hal_gpio_init(GPIOF,&init,GPIO_PIN_10);
    hal_gpio_init(GPIOE,&init,GPIO_PIN_8);
    
    
}
unsigned int i;
int find_command(const char* str)
{
    
    //串口中输入字符串,与结构体中每个元素的cmd_arr变量进行比较
    for(i=0;i<6;i++)
        if(my_strcmp(cmd_brr[i].cmd_arr,str))
            return i;
    return 9;
}
int main()
{
    hal_init();
    unsigned int index;
    char *string;
    while(1)
    {
        string=uart_get_string();
        index=find_command(string);
        if(index==9)
            uart_put_string("查找失败\n");
        else
            cmd_brr[index].gpio_write_pin(cmd_brr[index].gpiox,cmd_brr[index].pin,cmd_brr[index].status);
    }
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值