STM32开发之HC05基础驱动编写

2024-02-24 09:20

本文主要是介绍STM32开发之HC05基础驱动编写,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

驱动文件

头文件

//
// Created by shchl on 2024/2/22.
//#ifndef F4_PROJECT_BLE_HC05_H
#define F4_PROJECT_BLE_HC05_H#include "bsp.h"
#include "bsp_uart.h"typedef enum {hc05_resp_ok_status, /*接收正确*/hc05_resp_timeout_status, /*超时*/hc05_resp_fail_status /*失败*/
} hc05_resp_status; /*hc05响应状态*/void hc05_device_init(void);/*-----------------------------底层驱动需要实现的接口----------------------------------------*/extern void hc05_device_send(const char *cmd);extern char *hc05_device_wait_resp(void); /*等待设备响应数据*/
extern void hc05_device_enter_at_mode(void); /*进入AT模式*/
extern void hc05_device_quit_at_mode(void); /*退出AT模式*/
/*** 接收来自远端的数据(非阻塞)* @param dst* @return*/
extern uint16_t hc05_device_rec_remote_data(uint8_t *dst);
extern uint16_t hc05_device_send_remote_data(uint8_t *dst,uint16_t len);
/*---------------------------------------------------------------------*/void hc05_set_delay_cb(void (*delay_ms_call)(uint32_t));hc05_resp_status hc05_set_cmd(const char *cmd);char *hc05_query_cmd(const char *cmd);void hc05_delay_ms(uint32_t ms);hc05_resp_status hc05_check(uint8_t tryCnt);hc05_resp_status hc05_read_version(char *dst);hc05_resp_status hc05_read_work_stat(char *dst);hc05_resp_status hc05_read_addr(char *dst);hc05_resp_status hc05_read_name(char *dst);hc05_resp_status hc05_read_role(char *dst);hc05_resp_status hc05_read_pwd(char *dst);hc05_resp_status hc05_read_uart_cnf(char *dst);#endif //F4_PROJECT_BLE_HC05_H

源文件

//
// Created by shchl on 2024/2/22.
//#include "ble_hc05.h"static bool hc05_sub_str(char *dst, const char *src, char *prefix_str, char *suffix_str);static void (*hc05_delay_call)(uint32_t) =NULL;void hc05_set_delay_cb(void (*delay_ms_call)(uint32_t)) {hc05_delay_call = delay_ms_call;
}void hc05_delay_ms(uint32_t ms) {if (hc05_delay_call) {hc05_delay_call(ms);}
}/*** 设置指令* @param cmd*/
hc05_resp_status hc05_set_cmd(const char *cmd) {char *result = hc05_query_cmd(cmd);if (strstr(result, "OK")) {return hc05_resp_ok_status;} else if (strstr(result, "FAIL")) {return hc05_resp_fail_status;}return hc05_resp_timeout_status;}/*** 查询指令,并返回查询数据结果* @param cmd* @return*/
char *hc05_query_cmd(const char *cmd) {/*进入 AT模式*/hc05_device_enter_at_mode();hc05_delay_ms(10);hc05_device_send(cmd);//发送并接收/*退出 AT模式*/hc05_device_quit_at_mode();return hc05_device_wait_resp();}/*** @brief 蓝牙设备模块检测* @return*/
hc05_resp_status hc05_check(uint8_t tryCnt) {for (uint8_t i = 0; i < tryCnt; ++i) {if (hc05_set_cmd("AT\r\n") == hc05_resp_ok_status) {return hc05_resp_ok_status;}}return hc05_resp_timeout_status;
}/*** 获取蓝牙设备版本号* @param dst 保存位置* @return*/
hc05_resp_status hc05_read_version(char *dst) {char *ptr = hc05_query_cmd("AT+VERSION?\r\n"); /**/if (ptr) {if (hc05_sub_str(dst, ptr, "VERSION:", "\r\n")) {return hc05_resp_ok_status;}return hc05_resp_fail_status;}return hc05_resp_timeout_status;
}
hc05_resp_status hc05_read_work_stat(char *dst){char *ptr = hc05_query_cmd("AT+STATE?\r\n"); /**/if (ptr) {if (hc05_sub_str(dst, ptr, "STATE:", "\r\n")) {return hc05_resp_ok_status;}return hc05_resp_fail_status;}return hc05_resp_timeout_status;
}hc05_resp_status hc05_read_addr(char *dst){char *ptr = hc05_query_cmd("AT+ADDR?\r\n"); /**/if (ptr) {if (hc05_sub_str(dst, ptr, "ADDR:", "\r\n")) {return hc05_resp_ok_status;}return hc05_resp_fail_status;}return hc05_resp_timeout_status;
}hc05_resp_status hc05_read_name(char *dst){char *ptr = hc05_query_cmd("AT+NAME?\r\n"); /**/if (ptr) {if (hc05_sub_str(dst, ptr, "NAME:", "\r\n")) {return hc05_resp_ok_status;}return hc05_resp_fail_status;}return hc05_resp_timeout_status;
}hc05_resp_status hc05_read_role(char *dst){char *ptr = hc05_query_cmd("AT+ROLE?\r\n"); /**/if (ptr) {if (hc05_sub_str(dst, ptr, "ROLE:", "\r\n")) {return hc05_resp_ok_status;}return hc05_resp_fail_status;}return hc05_resp_timeout_status;
}hc05_resp_status hc05_read_pwd(char *dst){char *ptr = hc05_query_cmd("AT+PSWD?\r\n"); /**/if (ptr) {if (hc05_sub_str(dst, ptr, "PSWD:", "\r\n")) {return hc05_resp_ok_status;}return hc05_resp_fail_status;}return hc05_resp_timeout_status;
}hc05_resp_status hc05_read_uart_cnf(char *dst){char *ptr = hc05_query_cmd("AT+UART?\r\n"); /**/if (ptr) {if (hc05_sub_str(dst, ptr, "UART:", "\r\n")) {return hc05_resp_ok_status;}return hc05_resp_fail_status;}return hc05_resp_timeout_status;
}/*** @brief 提取字符串到指定位置* @param dst 保存子串* @param src 原始字符串* @param prefix_str 前缀(可以为NULL)* @param suffix_str 后缀(可以为NULL)* @return 是否提取成功*/
static bool hc05_sub_str(char *dst, const char *src, char *prefix_str, char *suffix_str) {const char *ptr_src = src;size_t sub_len; /*字串长度*/if (!ptr_src) return true;if (prefix_str != NULL) {ptr_src = strstr(ptr_src, prefix_str); /*查找前缀位置*/if (!ptr_src) return false;ptr_src += strlen(prefix_str); /**/}if (suffix_str) {char *endPtr = strstr(ptr_src, suffix_str);if (!endPtr) return false;sub_len = endPtr - ptr_src;} else {sub_len = strlen(ptr_src);}memcpy(dst, ptr_src, sub_len);dst[sub_len] = '\0'; /*添加结束符*/return true;
}

接口文件(对应硬件)

//
// Created by shchl on 2024/2/22.
//
#include "ble_hc05.h"#define HC05_AT_GPIO_PIN GPIO_PIN_9
#define HC05_AT_GPIO_PORT GPIOC
#define HC05_AT_GPIO_CLK_EN() __HAL_RCC_GPIOC_CLK_ENABLE()
#define hc05_uart_dev uart1_dev
static bool isCmdDatResp = false; /*是发送 指令数据标志位*/static void hc05_device_start_rec(void);static uint8_t hc05_rx_buf[UART1_RX_BUF_LEN];void hc05_device_init(void) {/*设置对应的延迟函数回调*/hc05_set_delay_cb(HAL_Delay);/*对应GPIO硬件初始化*/HC05_AT_GPIO_CLK_EN();GPIO_InitTypeDef GPIO_Init;GPIO_Init.Pin = HC05_AT_GPIO_PIN;GPIO_Init.Mode = GPIO_MODE_OUTPUT_PP;GPIO_Init.Pull = GPIO_PULLDOWN;GPIO_Init.Speed = GPIO_SPEED_MEDIUM;HAL_GPIO_Init(HC05_AT_GPIO_PORT, &GPIO_Init);/*对应串口硬件初始化*/uart1_dev_init(115200);hc05_device_start_rec();/*退出at模式*/hc05_device_quit_at_mode();}__weak void hc05_device_send(const char *cmd) {HAL_UART_Transmit(&hc05_uart_dev.uart_handle, (uint8_t *) cmd, strlen(cmd), 20);
}__weak char *hc05_device_wait_resp(void) {for (int i = 0; i < 10; ++i) {hc05_delay_ms(10);uint32_t len = CacheBuffer_Read_Data(hc05_uart_dev.rx_cache_ptr, hc05_rx_buf);if (len > 0) {hc05_rx_buf[len] = '\0';
//            printf("HC05_Rec_Str len:%lu,data:%s\r\n", len, hc05_rx_buf);return (char *) hc05_rx_buf;}}return NULL;
}__weak uint16_t hc05_device_rec_remote_data(uint8_t *dst) {hc05_device_quit_at_mode();return CacheBuffer_Read_Data(hc05_uart_dev.rx_cache_ptr, dst);
}uint16_t hc05_device_send_remote_data(uint8_t *dst,uint16_t len){hc05_device_quit_at_mode();HAL_UART_Transmit(&hc05_uart_dev.uart_handle, dst, len, 20);return len;
}
__weak void hc05_device_enter_at_mode(void) {HAL_GPIO_WritePin(HC05_AT_GPIO_PORT, HC05_AT_GPIO_PIN, GPIO_PIN_SET);
}__weak void hc05_device_quit_at_mode(void) {HAL_GPIO_WritePin(HC05_AT_GPIO_PORT, HC05_AT_GPIO_PIN, GPIO_PIN_RESET);
}/*** 串口函数回调中使用* @param Size*/
void HAL_UARTEx_RxEventCallback_HC05(uint16_t Size) {CacheBuffer_Update_Data_Len(hc05_uart_dev.rx_cache_ptr, Size);/*重新接收数据*/hc05_device_start_rec();
}static void hc05_device_start_rec(void) {HAL_UARTEx_ReceiveToIdle_DMA(&hc05_uart_dev.uart_handle,hc05_uart_dev.rx_cache_ptr->List[hc05_uart_dev.rx_cache_ptr->InIndex].start,CACHE_BUFFER_ONE_DATA_MAX_LEN);
}

串口驱动文件(针对stm32f4芯片,根据需要修改)

bsp_uart.h

//
// Created by shchl on 2024/2/21.
//#ifndef BSP_UART_H
#define BSP_UART_H#include "bsp.h"#include "cachebuffer.h"#define UART_PRINTF_DEV uart2_dev.uart_handle
/*-----------------------------------串口1配置----------------------------------------*/
#define UART1_DEV_RX_CACHE_ENABLE               1        /*串口1设备接收缓冲区使能*/
#define UART1_DEV_TX_CACHE_ENABLE               0         /*串口1设备发送缓冲区使能*/
#define UART1_IT_ENABLE                         1                   /*串口1中断*/#define UART1_RX_BUF_LEN                        1024
#define UART1_TX_BUF_LEN                        1024
/*-----------------------------------串口2配置----------------------------------------*/
#define UART2_DEV_RX_CACHE_ENABLE               0       /*串口2设备接收缓冲区使能*/
#define UART2_DEV_TX_CACHE_ENABLE               0       /*串口2设备发送缓冲区使能*/
#define UART2_IT_ENABLE                         0       /*串口2中断*/
#define UART2_RX_BUF_LEN                        1024
#define UART2_TX_BUF_LEN                        1024
typedef struct {CacheBuffer_t *rx_cache_ptr; /*接收缓存*/CacheBuffer_t *tx_cache_ptr; /*发送缓存*/UART_HandleTypeDef uart_handle;
} uart_dev_t; /*串口设备*/
extern uart_dev_t uart1_dev;
extern uart_dev_t uart2_dev;
void uart1_dev_init(uint32_t baud);void uart2_dev_init(uint32_t baud);#endif //BSP_UART_H

bsp_uart.c

//
// Created by shchl on 2024/2/21.
//#include "bsp_uart.h"static void uart_conf_init(UART_HandleTypeDef *uart_handle, uint32_t baud);uart_dev_t uart1_dev = {.uart_handle.Instance=USART1};
#if UART1_DEV_RX_CACHE_ENABLE
static CacheBuffer_t uart1_rx_cache;
static uint8_t uart1_rx_buf[UART1_RX_BUF_LEN] = {0};
#endif
#if UART1_DEV_TX_CACHE_ENABLE
static CacheBuffer_t uart1_tx_cache;
static uint8_t uart1_tx_buf[UART1_TX_BUF_LEN] = {0};
#endif/*** 重写串口printf* @param file* @param ptr* @param len* @return*/
int _write(int file, char *ptr, int len)
{(void)file;HAL_UART_Transmit(&UART_PRINTF_DEV, (const uint8_t *) ptr, len, 200);return len;
}
void uart1_dev_init(uint32_t baud) {
#if UART1_DEV_RX_CACHE_ENABLEuart1_dev.rx_cache_ptr = &uart1_rx_cache;CacheBuffer_Init(uart1_dev.rx_cache_ptr, uart1_rx_buf, UART1_RX_BUF_LEN);#endif#if UART1_DEV_TX_CACHE_ENABLEuart1_dev.tx_cache_ptr = &uart1_tx_cache;CacheBuffer_Init(uart1_dev.tx_cache_ptr, uart1_tx_buf, UART1_TX_BUF_LEN);
#endifuart_conf_init(&uart1_dev.uart_handle, baud);
}uart_dev_t uart2_dev = {.uart_handle.Instance=USART2};
#if UART2_DEV_RX_CACHE_ENABLE
static CacheBuffer_t uart2_rx_cache;
static uint8_t uart2_rx_buf[UART2_RX_BUF_LEN] = {0};
#endif
#if UART2_DEV_TX_CACHE_ENABLE
static CacheBuffer_t uart2_tx_cache;
static uint8_t uart2_tx_buf[UART2_TX_BUF_LEN] = {0};
#endif/*** @brief 串口2设备 初始化* @param baud*/
void uart2_dev_init(uint32_t baud) {
#if UART2_DEV_RX_CACHE_ENABLEuart2_dev.rx_cache_ptr = &uart2_rx_cache;CacheBuffer_Init(uart2_dev.rx_cache_ptr, uart2_rx_buf, UART2_RX_BUF_LEN);#endif#if UART2_DEV_TX_CACHE_ENABLEuart2_dev.tx_cache_ptr = &uart2_tx_cache;CacheBuffer_Init(uart2_dev.tx_cache_ptr, uart2_tx_buf, UART2_TX_BUF_LEN);
#endifuart_conf_init(&uart2_dev.uart_handle, baud);
}static void uart_conf_init(UART_HandleTypeDef *uart_handle, uint32_t baud) {uart_handle->Init.BaudRate = baud;uart_handle->Init.WordLength = UART_WORDLENGTH_8B;uart_handle->Init.StopBits = UART_STOPBITS_1;uart_handle->Init.Parity = UART_PARITY_NONE;uart_handle->Init.Mode = UART_MODE_TX_RX;uart_handle->Init.HwFlowCtl = UART_HWCONTROL_NONE;uart_handle->Init.OverSampling = UART_OVERSAMPLING_16;if (HAL_UART_Init(uart_handle) != HAL_OK) {Error_Handler();}
}

缓冲区文件(通用)

cachebuffer.h

//
// Created by shchl on 2024/2/21.
//#ifndef F4_DEMO_CACHEBUFFER_H
#define F4_DEMO_CACHEBUFFER_H#include <stdbool.h>
#include "stdio.h"
#include "string.h"#define CACHE_BUFFER_PARM_ASSERT(x)
#define CACHE_BUFFER_LIST_LEN 10 /*默认可以缓存10条数据,根据需要进行修改*/
#define CACHE_BUFFER_ONE_DATA_MAX_LEN 256 /*一条数据最大长度*/#define CACHE_BUFFER_MEMCPY memcpy
typedef struct {uint8_t *start;  /*起始位置*/uint8_t *end;   /*结束位置*/
} CacheIndexPtr_t; /*缓冲索引指针结构体*/typedef struct {uint32_t Count; /*累积缓存字节数*/uint16_t InIndex: 8; /*数据写入索引位置*/uint16_t OutIndex: 8; /*数据读出索引位置*/uint8_t *pBuf; /*实际数据指针存放位置*/uint32_t pBufCapacity; /*实际存放位置的容量*/CacheIndexPtr_t List[CACHE_BUFFER_LIST_LEN]; /*缓冲索引指针结构体集合*/
} CacheBuffer_t; /*缓冲Buffer结构体*/void CacheBuffer_Init(CacheBuffer_t *cacheBuffer, uint8_t *pool, int16_t size);void CacheBuffer_Update_Data_Len(CacheBuffer_t *cacheBuffer, uint32_t len);void CacheBuffer_Write_Data(CacheBuffer_t *cacheBuffer, uint8_t *pData, uint32_t len);uint32_t CacheBuffer_Read_Data(CacheBuffer_t *cacheBuffer, uint8_t *dst);#endif //F4_DEMO_CACHEBUFFER_H

cachebuffer.c

//
// Created by shchl on 2024/2/21.
//#include "cachebuffer.h"void CacheBuffer_Init(CacheBuffer_t *cacheBuffer, uint8_t *pool, int16_t size) {CACHE_BUFFER_PARM_ASSERT(cacheBuffer != NULL);CACHE_BUFFER_PARM_ASSERT(pool != NULL);cacheBuffer->pBufCapacity = size;cacheBuffer->InIndex = 0;cacheBuffer->OutIndex = 0;cacheBuffer->Count = 0;cacheBuffer->pBuf = pool;cacheBuffer->List[cacheBuffer->InIndex].start = pool;
}/*** @brief 缓存区更新数据长度,适用于缓冲区数据长度已更新,而数据长度未更新* @param cacheBuffer* @param len*/
void CacheBuffer_Update_Data_Len(CacheBuffer_t *cacheBuffer, uint32_t len) {cacheBuffer->Count += len;cacheBuffer->List[cacheBuffer->InIndex].end = cacheBuffer->pBuf + cacheBuffer->Count - 1;cacheBuffer->InIndex++;if (cacheBuffer->InIndex == CACHE_BUFFER_LIST_LEN) {/*判断是否已经到了最后的位置*/cacheBuffer->InIndex = 0;}/*判断是否需要覆盖数据*/if (cacheBuffer->pBufCapacity - cacheBuffer->Count < CACHE_BUFFER_ONE_DATA_MAX_LEN) {cacheBuffer->Count = 0;cacheBuffer->List[cacheBuffer->InIndex].start = cacheBuffer->pBuf;} else {cacheBuffer->List[cacheBuffer->InIndex].start = cacheBuffer->pBuf + cacheBuffer->Count;}
}void CacheBuffer_Write_Data(CacheBuffer_t *cacheBuffer, uint8_t *pData, uint32_t len) {/*确定写入数据位置*/if (cacheBuffer->pBufCapacity - cacheBuffer->Count < len) {/*判断写入数据是否大于剩下空间*/cacheBuffer->Count = 0;cacheBuffer->List[cacheBuffer->InIndex].start = cacheBuffer->pBuf;} else {cacheBuffer->List[cacheBuffer->InIndex].start = cacheBuffer->pBuf + cacheBuffer->Count;}CACHE_BUFFER_MEMCPY(cacheBuffer->List[cacheBuffer->InIndex].start, pData, len);cacheBuffer->Count += len;cacheBuffer->List[cacheBuffer->InIndex].end = cacheBuffer->pBuf + cacheBuffer->Count - 1;cacheBuffer->InIndex++;if (cacheBuffer->InIndex == CACHE_BUFFER_LIST_LEN) {/*判断是否已经到了最后的位置*/cacheBuffer->InIndex = 0;}
}uint32_t CacheBuffer_Read_Data(CacheBuffer_t *cacheBuffer, uint8_t *dst) {//判断是否有数据CacheIndexPtr_t *ptr;uint32_t len = 0;if (cacheBuffer->OutIndex != cacheBuffer->InIndex) {// 有数据ptr = &cacheBuffer->List[cacheBuffer->OutIndex];len = ptr->end - ptr->start + 1;CACHE_BUFFER_MEMCPY(dst, ptr->start, len);cacheBuffer->OutIndex++;if (cacheBuffer->OutIndex == CACHE_BUFFER_LIST_LEN) {cacheBuffer->OutIndex = 0; /*标记到最后位置*/}}return len;
}

这篇关于STM32开发之HC05基础驱动编写的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/741657

相关文章

使用DeepSeek API 结合VSCode提升开发效率

《使用DeepSeekAPI结合VSCode提升开发效率》:本文主要介绍DeepSeekAPI与VisualStudioCode(VSCode)结合使用,以提升软件开发效率,具有一定的参考价值... 目录引言准备工作安装必要的 VSCode 扩展配置 DeepSeek API1. 创建 API 请求文件2.

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做

Java中的Opencv简介与开发环境部署方法

《Java中的Opencv简介与开发环境部署方法》OpenCV是一个开源的计算机视觉和图像处理库,提供了丰富的图像处理算法和工具,它支持多种图像处理和计算机视觉算法,可以用于物体识别与跟踪、图像分割与... 目录1.Opencv简介Opencv的应用2.Java使用OpenCV进行图像操作opencv安装j

使用PyQt5编写一个简单的取色器

《使用PyQt5编写一个简单的取色器》:本文主要介绍PyQt5搭建的一个取色器,一共写了两款应用,一款使用快捷键捕获鼠标附近图像的RGB和16进制颜色编码,一款跟随鼠标刷新图像的RGB和16... 目录取色器1取色器2PyQt5搭建的一个取色器,一共写了两款应用,一款使用快捷键捕获鼠标附近图像的RGB和16

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

C#图表开发之Chart详解

《C#图表开发之Chart详解》C#中的Chart控件用于开发图表功能,具有Series和ChartArea两个重要属性,Series属性是SeriesCollection类型,包含多个Series对... 目录OverviChina编程ewSeries类总结OverviewC#中,开发图表功能的控件是Char

使用Java编写一个文件批量重命名工具

《使用Java编写一个文件批量重命名工具》这篇文章主要为大家详细介绍了如何使用Java编写一个文件批量重命名工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录背景处理1. 文件夹检查与遍历2. 批量重命名3. 输出配置代码片段完整代码背景在开发移动应用时,UI设计通常会提供不