当前位置: 首页 > news >正文

2014 网站建设公司运营策划方案

2014 网站建设,公司运营策划方案,旅游公司网站建设策划书,企业网站怎样做一、空闲中断 STM32的串口具有空闲中断,什么叫做空闲呢?如何触发空闲中断呢? 空闲:串口发送的两个字符之间间隔非常短,所以在两个字符之间不叫空闲。空闲的定义是总线上在一个字节的时间内没有再接收到数据。触发条件…

一、空闲中断

STM32的串口具有空闲中断,什么叫做空闲呢?如何触发空闲中断呢?

  • 空闲:串口发送的两个字符之间间隔非常短,所以在两个字符之间不叫空闲。空闲的定义是总线上在一个字节的时间内没有再接收到数据。
  • 触发条件:空闲中断是检测到有数据被接收后,总线上在一个字节的时间内没有再接收到数据的时候发生的。而总线在什么情况时,会有一个字节时间内没有接收到数据呢?一般就只有一个数据帧发送完成的情况,所以串口的空闲中断也叫帧中断。

开启空闲中断后,要重写对应的回调函数HAL_UARTEx_RxEventCallback,在函数里做些处理。

二、实验内容

使用USART1外设,接收电脑发来的数据,然后将接收到的数据发送给电脑。
在这里插入图片描述

二、STM32CUBEMX配置

USART1的模式为异步通信115200波特率数据长度8位无校验位停止位1位
在这里插入图片描述
DMA Settings的配置,开启串口接收的DMA
在这里插入图片描述
NVIC Settings的配置,开启USART1的全局中断。
在这里插入图片描述
这里两个中断优先级我都给了1,根据不同情况修改中断优先级
在这里插入图片描述

三、keil代码

首先确保魔术棒中的Use MicroLIB这个选项勾选上,不然串口发送数据会不正常
在这里插入图片描述
添加头文件,因为要使用memset函数

#include "string.h"

定义串口接收数据数组

#define BUFF_SIZE	128			    //接收缓存大小
uint8_t rx_buffer[BUFF_SIZE];      // 创建接收缓存,大小为BUFF_SIZE

手动在usart.h外部声明hdma_usart1_rx,在main函数中要使用

extern UART_HandleTypeDef huart1;/* USER CODE BEGIN Private defines */
extern DMA_HandleTypeDef hdma_usart1_rx;        //手动外部声明
/* USER CODE END Private defines */void MX_USART1_UART_Init(void);

main函数初始化添加这两个函数,不然串口首次无法进入中断

HAL_UARTEx_ReceiveToIdle_DMA(&huart1,rx_buffer,BUFF_SIZE);	//手动开启串口DMA模式接收数据
__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);		   	//手动关闭DMA_IT_HT中断

重定向HAL_UARTEx_RxEventCallback串口接收完成回调函数

/* 串口接收完成回调函数 */
void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size)
{if (huart->Instance == USART1){HAL_UARTEx_ReceiveToIdle_DMA(&huart1, rx_buffer, BUFF_SIZE); // 接收完毕后重启串口DMA模式接收数据HAL_UART_Transmit(&huart1, rx_buffer, Size, 0xffff);         // 将接收到的数据再发出__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);		   	// 手动关闭DMA_IT_HT中断memset(rx_buffer, 0, BUFF_SIZE);							// 清除接收缓存}
}	

重定向HAL_UART_ErrorCallback串口错误回调函数

/* 串口错误回调函数 */
void HAL_UART_ErrorCallback(UART_HandleTypeDef * huart)
{if(huart->Instance == USART1){HAL_UARTEx_ReceiveToIdle_DMA(&huart1, rx_buffer, BUFF_SIZE); //手动开启串口DMA模式接收数据__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);		   // 手动关闭DMA_IT_HT中断memset(rx_buffer, 0, BUFF_SIZE);							   // 清除接收缓存}
}

以下是main函数完整代码

#include "main.h"
#include "dma.h"
#include "usart.h"
#include "gpio.h"#include "stdio.h"
#include "string.h"void SystemClock_Config(void);#define BUFF_SIZE	128			//接收缓存大小
uint8_t rx_buffer[BUFF_SIZE];  // 创建接收缓存,大小为BUFF_SIZEint main(void)
{HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_DMA_Init();MX_USART1_UART_Init();HAL_UARTEx_ReceiveToIdle_DMA(&huart1,rx_buffer,BUFF_SIZE);	//手动开启串口DMA模式接收数据__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);		   	//手动关闭DMA_IT_HT中断	while (1){}
}
void SystemClock_Config(void)
{//...
}
/* USER CODE BEGIN 4 */
/* 串口接收完成回调函数 */
void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size)
{if (huart->Instance == USART1){HAL_UARTEx_ReceiveToIdle_DMA(&huart1, rx_buffer, BUFF_SIZE); /// 接收完毕后重启串口DMA模式接收数据HAL_UART_Transmit(&huart1, rx_buffer, Size, 0xffff);         // 将接收到的数据再发出__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);		   		// 手动关闭DMA_IT_HT中断memset(rx_buffer, 0, BUFF_SIZE);							   	// 清除接收缓存}
}
/* 串口错误回调函数 */
void HAL_UART_ErrorCallback(UART_HandleTypeDef * huart)
{if(huart->Instance == USART1){HAL_UARTEx_ReceiveToIdle_DMA(&huart1, rx_buffer, BUFF_SIZE);//手动开启串口DMA模式接收数据__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);		   // 手动关闭DMA_IT_HT中断memset(rx_buffer, 0, BUFF_SIZE);							   // 清除接收缓存}
}
/* USER CODE END 4 */
void Error_Handler(void)
{/* USER CODE BEGIN Error_Handler_Debug *//* User can add his own implementation to report the HAL error return state */__disable_irq();while (1){}/* USER CODE END Error_Handler_Debug */
}

四、原理讲解

我们使用串口空闲中断的目的是为了获取完整的数据,并且是由软件自动判断是否接收完整。当软件判断接收到完整的数据时,就会产生中断进入回调函数HAL_UARTEx_RxEventCallback
软件判断接收到完整数据有两种情况:

  • 1、数据接收后,一个字节未接收到数据;
  • 2、当前的数据接收长度预设接收长度相等;预设接收长度HAL_UARTEx_ReceiveToIdle_DMA的参数BUFF_SIZE

预设接收长度要考虑好,若数据接收长度预设接受长度,就会出现数据的丢失情况。因为数据接收长度是随着接收不断累加的,其大小等于预设接收长度时就会触发中断,程序就会判断成接收到完整数据,就是第二情况。

1、HAL_UARTEx_ReceiveToIdle_DMA

main函数里我们调用了HAL_UARTEx_ReceiveToIdle_DMA,该函数会开启USART1的空闲中断,并启用串口以DMA方式接收数据,当数据接收完成之后进入HAL_UARTEx_RxEventCallback回调函数。随后,我们在回调函数里进行数据处理即可。

2、__HAL_DMA_DISABLE_IT

为什么要手动关闭这个中断,如果不关闭这个中断,程序在接收一串完整的数据时会进入执行两次HAL_UARTEx_RxEventCallback。一次是数据传输一半时,一次数数据传输完成时,这与我们预期的不符。我们只希望在数据传输完成时进入回调函数进行数据处理。

__HAL_DMA_DISABLE_IT的回调函数是UART_DMARxHalfCplt,该回调函数会执行一次HAL_UARTEx_RxEventCallback

3、HAL_UARTEx_RxEventCallback

在该回调函数里,需要手动开启中断,才能进入下一次的中断。

4、HAL_UART_ErrorCallback

重写这个回调函数主要是为了防止一些错误情况的发生。若产生错误中断进入HAL_UART_ErrorCallback,则无法进入HAL_UARTEx_RxEventCallback回调函数,也就无法开启下一次空闲中断。所以,这里在HAL_UART_ErrorCallback里手动开启了空闲中断,做了一些恢复处理。

错误示例:开发板设置115200波特率,电脑串口用9600波特率,电脑发送数据之后,程序会进入串口错误中断,而进入不了空闲中断。若程序当中未在错误中断进行错误处理,即使电脑串口修改成115200波特率进行通信,程序也无法进入空闲中断。

五、参考文章

配置和代码链接,里面详解了相关一些函数的源码
链接: 【STM32 HAL库实战】串口DMA + 空闲中断 实现不定长数据接收
空闲中断讲解
链接: STM32串口之空闲中断

http://www.hengruixuexiao.com/news/39524.html

相关文章:

  • 慈溪做无痛同济 amp 网站企业推广网络营销外包服务
  • 南宁建站免费模板西安百度推广怎么做
  • 深圳网站建设公司制作定制北京aso优化
  • 做网站需要什么材料seo排名优化推荐
  • seo技术教程在线咨询做网站建设优化的公司排名
  • 黑客网站装b免费的网页模板网站
  • 重庆建网站多少钱希爱力双效片
  • 中怎么做网站上下载图片的功能网站的seo
  • 专业网站制作公司招聘百度权重什么意思
  • 公司内部网站怎么做百度搜索工具
  • 无锡公共工程建设中心网站最新军事战争新闻消息
  • cms系统和网站后台系统企业网络营销方案策划
  • wordpress采集网页文章石家庄seo网络推广
  • 荆州 网站建设设计公司网站模板
  • 昆明网站seo多少钱seo人员培训
  • 哪家app定制开发好深圳seo优化公司排名
  • 广州响应式网站广州seo服务
  • 做的网站一模一样会被告吗百度快照优化排名推广
  • 腾讯云网站建设外链工具软件
  • 做网站的视频360收录批量查询
  • 网站建设与管理资料下载电商自学网
  • 怎么做私服发布网站seo是什么意思新手怎么做seo
  • 商城网站开发项目实战java公司网站建设费用多少
  • 网站模板提供源码seo收费还是免费
  • dedecms企业网站模板seo网站优化外包
  • 如何完善网站建设如何推广店铺呢
  • php毕业设计二手网站怎么做软文营销模板
  • 中象做网站怎么样seo是什么意思?
  • wordpress $user_id网页优化方法
  • wordpress点赞打赏上海优化外包公司排名