如何查看dma空闲中断接收了多少数
时间: 2023-05-09 17:02:49 浏览: 40
要查看DMA空闲中断接收了多少数,可以采取以下步骤:
1. 首先,需要打开DMA控制器的空闲中断开关,使得当DMA通道状态为IDLE时会触发DMA空闲中断。不同芯片的DMA控制器实现方式不同,可以参考芯片手册的相关章节进行设置。
2. 接下来,在空闲中断的中断服务函数中,可以通过读取DMA控制器的状态寄存器或者空闲中断寄存器来获取DMA空闲中断接收了多少数。具体的寄存器名称和位定义也需要参考芯片手册。
3. 最后,可以将获取到的DMA空闲中断接收数输出到调试串口或者通过其他方式进行展示,方便我们进行统计和分析。
需要注意的是,不同芯片的DMA控制器可能对于空闲中断的处理方式有些许不同,例如一些芯片可能需要清除空闲中断标志位等操作,因此具体的实现方式需要参考对应的芯片手册。
相关问题
HAL库串口+DMA空闲中断接收不定长数据
使用DMA空闲中断接收不定长数据的步骤如下:
1. 初始化串口和DMA:配置串口为接收模式,启用DMA传输;配置DMA为循环模式,传输大小为1字节,传输方向为从串口接收数据到内存。
2. 开启DMA传输:调用DMA启动函数启动DMA传输。
3. 开启串口接收中断:调用串口中断使能函数,开启空闲中断。
4. 在空闲中断中处理数据:当DMA传输完成并且串口没有接收到新数据时,说明接收完成,可以在空闲中断中处理接收到的数据。
下面是一个简单的例子:
```c
#include "stm32f4xx_hal.h"
#define RX_BUF_SIZE 256
UART_HandleTypeDef huart1;
DMA_HandleTypeDef hdma_usart1_rx;
uint8_t rx_buf[RX_BUF_SIZE];
uint8_t rx_len = 0;
uint8_t rx_flag = 0;
void UART_Init(void)
{
huart1.Instance = USART1;
huart1.Init.BaudRate = 115200;
huart1.Init.WordLength = UART_WORDLENGTH_8B;
huart1.Init.StopBits = UART_STOPBITS_1;
huart1.Init.Parity = UART_PARITY_NONE;
huart1.Init.Mode = UART_MODE_RX;
huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;
huart1.Init.OverSampling = UART_OVERSAMPLING_16;
if (HAL_UART_Init(&huart1) != HAL_OK)
{
Error_Handler();
}
hdma_usart1_rx.Instance = DMA2_Stream2;
hdma_usart1_rx.Init.Channel = DMA_CHANNEL_4;
hdma_usart1_rx.Init.Direction = DMA_PERIPH_TO_MEMORY;
hdma_usart1_rx.Init.PeriphInc = DMA_PINC_DISABLE;
hdma_usart1_rx.Init.MemInc = DMA_MINC_ENABLE;
hdma_usart1_rx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;
hdma_usart1_rx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;
hdma_usart1_rx.Init.Mode = DMA_CIRCULAR;
hdma_usart1_rx.Init.Priority = DMA_PRIORITY_LOW;
hdma_usart1_rx.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
if (HAL_DMA_Init(&hdma_usart1_rx) != HAL_OK)
{
Error_Handler();
}
__HAL_LINKDMA(&huart1, hdmarx, hdma_usart1_rx);
HAL_NVIC_SetPriority(DMA2_Stream2_IRQn, 0, 0);
HAL_NVIC_EnableIRQ(DMA2_Stream2_IRQn);
HAL_UART_Receive_DMA(&huart1, rx_buf, RX_BUF_SIZE);
}
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
if (huart == &huart1)
{
rx_len += RX_BUF_SIZE - hdma_usart1_rx.Instance->NDTR;
HAL_UART_Receive_DMA(huart, rx_buf, RX_BUF_SIZE);
}
}
void HAL_UART_RxHalfCpltCallback(UART_HandleTypeDef *huart)
{
if (huart == &huart1)
{
rx_len += RX_BUF_SIZE / 2 - hdma_usart1_rx.Instance->NDTR;
}
}
void HAL_UART_ErrorCallback(UART_HandleTypeDef *huart)
{
if (huart == &huart1)
{
__HAL_UART_CLEAR_PEFLAG(&huart1);
HAL_UART_Receive_DMA(huart, rx_buf, RX_BUF_SIZE);
}
}
void DMA2_Stream2_IRQHandler(void)
{
HAL_DMA_IRQHandler(&hdma_usart1_rx);
}
void HAL_UART_IdleCallback(UART_HandleTypeDef *huart)
{
if (huart == &huart1)
{
if (hdma_usart1_rx.Instance->NDTR == RX_BUF_SIZE)
{
rx_len = RX_BUF_SIZE;
rx_flag = 1;
}
else
{
rx_len = RX_BUF_SIZE - hdma_usart1_rx.Instance->NDTR;
}
}
}
int main(void)
{
HAL_Init();
UART_Init();
while (1)
{
if (rx_flag)
{
// 处理接收到的数据
rx_flag = 0;
}
}
}
```
在上面的例子中,我们使用了循环DMA传输模式,当接收到一定数量的数据后,将触发空闲中断,并在空闲中断中处理接收到的数据。同时,在DMA传输完成和空闲中断中,我们使用了两个不同的回调函数,分别处理DMA传输完成和空闲中断的事件。
串口空闲中断 +DMA中断接收
串口空闲中断和DMA中断接收是两种不同的接收方式。
串口空闲中断接收是指当串口接收到数据后,数据传输完成并且串口空闲时,触发中断来进行数据处理。这种方式比较简单,适用于数据量较小的情况。
DMA中断接收是指使用DMA控制器来进行数据传输,当DMA传输完成后触发中断进行数据处理。这种方式适用于数据量较大的情况,可以提高数据传输的效率。
下面是一个基于STM32的串口空闲中断+DMA中断接收的例子:
```c
#include "stm32f4xx.h"
#include "stm32f4xx_conf.h"
#define USARTx USART2
#define DMAx DMA1
#define DMAx_Streamx DMA1_Stream5
#define DMAx_Streamx_IRQn DMA1_Stream5_IRQn
#define DMAx_Streamx_IRQHandler DMA1_Stream5_IRQHandler
#define BUFFER_SIZE 1024
uint8_t buffer[BUFFER_SIZE];
volatile uint16_t bufferIndex = 0;
void initUSART()
{
USART_InitTypeDef USART_InitStruct;
GPIO_InitTypeDef GPIO_InitStruct;
NVIC_InitTypeDef NVIC_InitStruct;
DMA_InitTypeDef DMA_InitStruct;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
GPIO_InitStruct.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3;
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF;
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStruct.GPIO_OType = GPIO_OType_PP;
GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_UP;
GPIO_Init(GPIOA, &GPIO_InitStruct);
GPIO_PinAFConfig(GPIOA, GPIO_PinSource2, GPIO_AF_USART2);
GPIO_PinAFConfig(GPIOA, GPIO_PinSource3, GPIO_AF_USART2);
USART_InitStruct.USART_BaudRate = 115200;
USART_InitStruct.USART_WordLength = USART_WordLength_8b;
USART_InitStruct.USART_StopBits = USART_StopBits_1;
USART_InitStruct.USART_Parity = USART_Parity_No;
USART_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
USART_InitStruct.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
USART_Init(USARTx, &USART_InitStruct);
USART_ITConfig(USARTx, USART_IT_IDLE, ENABLE);
DMA_InitStruct.DMA_Channel = DMA_Channel_4;
DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralToMemory;
DMA_InitStruct.DMA_Memory0BaseAddr = (uint32_t)buffer;
DMA_InitStruct.DMA_BufferSize = BUFFER_SIZE;
DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&USARTx->DR;
DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStruct.DMA_Mode = DMA_Mode_Circular;
DMA_InitStruct.DMA_Priority = DMA_Priority_Medium;
DMA_InitStruct.DMA_FIFOMode = DMA_FIFOMode_Disable;
DMA_InitStruct.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
DMA_InitStruct.DMA_MemoryBurst = DMA_MemoryBurst_Single;
DMA_InitStruct.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
DMA_Init(DMAx_Streamx, &DMA_InitStruct);
NVIC_InitStruct.NVIC_IRQChannel = DMAx_Streamx_IRQn;
NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStruct);
USART_Cmd(USARTx, ENABLE);
DMA_Cmd(DMAx_Streamx, ENABLE);
}
void DMAx_Streamx_IRQHandler()
{
if (DMA_GetITStatus(DMAx_Streamx, DMA_IT_TCIF5) != RESET)
{
DMA_ClearITPendingBit(DMAx_Streamx, DMA_IT_TCIF5);
DMA_Cmd(DMAx_Streamx, DISABLE);
bufferIndex = BUFFER_SIZE - DMA_GetCurrDataCounter(DMAx_Streamx);
}
}
void USART2_IRQHandler()
{
if (USART_GetITStatus(USART2, USART_IT_IDLE) != RESET)
{
USART_ClearITPendingBit(USART2, USART_IT_IDLE);
DMAx_Streamx->CR &= ~(1 << 0); // Disable DMA Stream
DMA_ClearFlag(DMAx_Streamx, DMA_FLAG_TCIF5 | DMA_FLAG_HTIF5 | DMA_FLAG_TEIF5 | DMA_FLAG_DMEIF5 | DMA_FLAG_FEIF5);
DMA_Cmd(DMAx_Streamx, ENABLE);
}
}
int main()
{
initUSART();
while (1)
{
if (bufferIndex > 0)
{
// 处理接收到的数据
bufferIndex = 0;
}
}
}
```
该例子中使用DMA控制器进行数据传输,并通过串口空闲中断触发DMA传输完成中断。在DMA传输完成中断中,通过DMA_GetCurrDataCounter函数获取当前未传输的数据长度,即接收到的数据长度。在串口空闲中断中,通过禁用并重新启用DMA Stream来触发DMA传输完成中断。
相关推荐
![rar](https://img-home.csdnimg.cn/images/20210720083606.png)
![zip](https://img-home.csdnimg.cn/images/20210720083736.png)
![rar](https://img-home.csdnimg.cn/images/20210720083606.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)