久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费
標題:
STM32F103 UART1 DMA不定長收發源碼
[打印本頁]
作者:
annko2005
時間:
2018-12-12 17:14
標題:
STM32F103 UART1 DMA不定長收發源碼
在使用STM32F103做串口收發時,如果采用中斷方式可能會丟包,但是采用DMA方式就可以避免這種問題,案例就是用DMA方式傳輸數據
DMA是一種不使用CPU而將數據從一片地址空間復制到另一片地址空間的總線,這樣就減少了CPU的負擔,使其能夠更加專注于數據運算。
為了能夠減少CPU的負擔,DMA應該采取中斷方式而非查詢模式。
但是非常不幸的是,STM32F103只為DMA提供了三種中斷:半步中斷、完成中斷和錯誤中斷。
如果UART接收的是定幀長的數據,則可以開啟DMA半步中斷,并且目標地址長度為幀長兩倍。這樣每接收完一幀進一次中斷,進行某些操作,是很理想的。
然而當遇到如同GPS一樣不定幀長的數據時,如果仍用半步中斷則難以確定目標地址的長度。
所以需要放棄使用DMA的中斷,轉而使用的是另一種比較特別的中斷:UART空閑中斷。
先來介紹一下UART空閑中斷。
UART常用的接收中斷響應有:接收數據 就緒可讀中斷RXNE(這是最常用的)、數據溢出中斷(ORE)、奇偶校驗錯中斷(PE)和空閑中斷(IDLE)。
空閑中斷是指當總線檢測到一幀發完后, 總線空閑則會將此位置一,如果USART_CR1中的IDLEIE為’1’,則產生中斷。
空閑中斷有兩個比較有意思的特點:
1、清零方式:軟件清零,先讀USART_SR,然后讀USART_DR
2、直到RXNE被置一后IDLE才能被重讀置一,即IDLE被置一并軟件清零后,只有之后再次接收到數據,IDLE才能被置一。
這樣就防止了總線長時間空閑而多次引發空閑中斷。
單片機源程序如下:
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "stm32f1xx_hal.h"
#include "dma.h"
#include "tim.h"
#include "usart.h"
#include "gpio.h"
/* USER CODE BEGIN Includes */
#include "stdarg.h"
#include "string.h"
//#include <stdlib.h>
#ifdef __GNUC__
/* With GCC/RAISONANCE, small printf (option LD Linker->Libraries->Small printf
set to 'Yes') calls __io_putchar() */
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif /* __GNUC__ */
/* USER CODE END Includes */
/* Private variables ---------------------------------------------------------*/
/* USER CODE BEGIN PV */
extern UART_HandleTypeDef huart1;
extern DMA_HandleTypeDef hdma_usart1_rx;
extern DMA_HandleTypeDef hdma_usart1_tx;
/* Private variables ---------------------------------------------------------*/
#define BUFFER_SIZE 100
uint8_t rx_len=0; //接收一幀數據的長度
uint8_t recv_end_flag=0; //一幀數據接收完成標志
uint8_t rx_buffer[BUFFER_SIZE]={"123"}; //接收數據緩存
uint8_t tx_buffer[BUFFER_SIZE];
/* USER CODE END PV */
/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */
/* Private function prototypes -----------------------------------------------*/
/* USER CODE END PFP */
/* USER CODE BEGIN 0 */
/* USER CODE END 0 */
/**
* @brief The application entry point.
*
* @retval None
*/
int main(void)
{
/* USER CODE BEGIN 1 */
/* USER CODE END 1 */
/* MCU Configuration----------------------------------------------------------*/
/* Reset of all peripherals, Initializes the Flash interface and the Systick. */
HAL_Init();
/* USER CODE BEGIN Init */
/* USER CODE END Init */
/* Configure the system clock */
SystemClock_Config();
/* USER CODE BEGIN SysInit */
/* USER CODE END SysInit */
/* Initialize all configured peripherals */
MX_GPIO_Init();
MX_DMA_Init();
MX_TIM2_Init();
MX_USART1_UART_Init();
/* USER CODE BEGIN 2 */
//printf("tst uart DMA begin\r\n");
//__HAL_UART_ENABLE_IT(&huart1, UART_IT_RXNE);
__HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE); //使能IDLE中斷
HAL_UART_Receive_DMA(&huart1,rx_buffer,BUFFER_SIZE);//啟動DMA接收
/* USER CODE END 2 */
/* Infinite loop */
/* USER CODE BEGIN WHILE */
while (1)
{
HAL_GPIO_WritePin(Tst_Pin_GPIO_Port, Tst_Pin_Pin, GPIO_PIN_RESET);
if(recv_end_flag ==1)
{
HAL_UART_Transmit(&huart1,rx_buffer, rx_len,200);//接收數據打印出來
rx_len=0;
recv_end_flag=0;
//memset(rx_buffer,0x00,BUFFER_SIZE);
HAL_UART_Receive_DMA(&huart1,rx_buffer,BUFFER_SIZE); //重啟DMA接收
}
HAL_GPIO_WritePin(Tst_Pin_GPIO_Port, Tst_Pin_Pin, GPIO_PIN_SET);
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
/* USER CODE END 3 */
}
}
/**
* @brief System Clock Configuration
* @retval None
*/
void SystemClock_Config(void)
{
RCC_OscInitTypeDef RCC_OscInitStruct;
RCC_ClkInitTypeDef RCC_ClkInitStruct;
/**Initializes the CPU, AHB and APB busses clocks
*/
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI;
RCC_OscInitStruct.HSIState = RCC_HSI_ON;
RCC_OscInitStruct.HSICalibrationValue = 16;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
{
_Error_Handler(__FILE__, __LINE__);
}
/**Initializes the CPU, AHB and APB busses clocks
*/
RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI;
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK)
{
_Error_Handler(__FILE__, __LINE__);
}
HAL_RCC_MCOConfig(RCC_MCO, RCC_MCO1SOURCE_SYSCLK, RCC_MCODIV_1);
/**Configure the Systick interrupt time
*/
HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq()/1000);
/**Configure the Systick
*/
HAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK);
/* SysTick_IRQn interrupt configuration */
HAL_NVIC_SetPriority(SysTick_IRQn, 0, 0);
}
/* USER CODE BEGIN 4 */
void UART_IDLE_Callback(UART_HandleTypeDef *huart)
{
uint16_t i = 0;
if((__HAL_UART_GET_FLAG(huart,UART_FLAG_IDLE) != RESET))
{
if(huart->Instance == USART1)
{
__HAL_UART_CLEAR_IDLEFLAG(huart);
i = huart->Instance->SR;
i = huart->Instance->DR;
i = hdma_usart1_rx.Instance->CNDTR; //讀取DMA剩余傳輸數量
HAL_UART_DMAStop(huart);
/* 此處處理數據,主要是拷貝和置位標志位 */
if((recv_end_flag == 0)&&(i!=BUFFER_SIZE))
{
rx_len = BUFFER_SIZE - i;
//memcpy(tx_buffer,rx_buffer,rx_len);
recv_end_flag = 1;
}
else
{
/* 清空緩存,重新接收 */
//memset(rx_buffer,0x00,BUFFER_SIZE);
HAL_UART_Receive_DMA(huart,(uint8_t *)&rx_buffer,BUFFER_SIZE);
}
}
}
}
/* USER CODE END 4 */
/**
* @brief This function is executed in case of error occurrence.
* @param file: The file name as string.
* @param line: The line in file as a number.
* @retval None
*/
void _Error_Handler(char *file, int line)
{
/* USER CODE BEGIN Error_Handler_Debug */
/* User can add his own implementation to report the HAL error return state */
while(1)
{
}
/* USER CODE END Error_Handler_Debug */
}
#ifdef USE_FULL_ASSERT
/**
* @brief Reports the name of the source file and the source line number
* where the assert_param error has occurred.
* @param file: pointer to the source file name
* @param line: assert_param error line source number
* @retval None
*/
void assert_failed(uint8_t* file, uint32_t line)
{
/* USER CODE BEGIN 6 */
/* User can add his own implementation to report the file name and line number,
ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
/* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */
/**
* @}
*/
/**
* @}
*/
/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/
復制代碼
所有資料51hei提供下載:
103_UARTDMA.rar
(445.2 KB, 下載次數: 104)
2018-12-13 04:00 上傳
點擊文件名下載附件
UART DMA
下載積分: 黑幣 -5
歡迎光臨 (http://www.zg4o1577.cn/bbs/)
Powered by Discuz! X3.1
主站蜘蛛池模板:
日本电影韩国电影免费观看
|
色婷婷久久久亚洲一区二区三区
|
在线看免费
|
久久国产精品免费
|
亚洲情视频
|
国产精品一码二码三码在线
|
国产精品一区二区久久精品爱微奶
|
www.99热这里只有精品
|
欧美一区二区成人
|
伊人网一区
|
夜夜草导航
|
国产精品久久久久久久久久软件
|
hdfreexxxx中国妞
|
91精品无人区卡一卡二卡三
|
伊人网影院
|
激情五月婷婷综合
|
久久国内精品
|
久草热线
|
av一级一片
|
在线观看黄免费
|
日韩在线一区二区
|
k8久久久一区二区三区
|
国产一区二区精华
|
色视频网站
|
日本精品视频一区二区
|
欧美不卡一区
|
亚洲视频二区
|
自拍在线
|
成人午夜精品
|
日日夜夜天天
|
国产一区不卡
|
欧美11一13sex性hd
|
久久久精品一区
|
亚洲国产激情
|
免费看黄色视屏
|
国产激情三区
|
国产剧情一区二区三区
|
亚洲国产区
|
久久精品中文
|
国产成人福利在线观看
|
成人午夜高清
|