久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 5389|回復: 0
收起左側

STM32F103 UART1 DMA不定長收發源碼

[復制鏈接]
ID:81176 發表于 2018-12-12 17:14 | 顯示全部樓層 |閱讀模式
在使用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才能被置一。
   這樣就防止了總線長時間空閑而多次引發空閑中斷。

單片機源程序如下:
  1. /* Includes ------------------------------------------------------------------*/
  2. #include "main.h"
  3. #include "stm32f1xx_hal.h"
  4. #include "dma.h"
  5. #include "tim.h"
  6. #include "usart.h"
  7. #include "gpio.h"

  8. /* USER CODE BEGIN Includes */
  9. #include "stdarg.h"
  10. #include "string.h"
  11. //#include <stdlib.h>
  12. #ifdef __GNUC__

  13.   /* With GCC/RAISONANCE, small printf (option LD Linker->Libraries->Small printf
  14. set to 'Yes') calls __io_putchar() */
  15. #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
  16. #else

  17.   #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
  18. #endif /* __GNUC__ */
  19. /* USER CODE END Includes */

  20. /* Private variables ---------------------------------------------------------*/

  21. /* USER CODE BEGIN PV */
  22. extern UART_HandleTypeDef huart1;
  23. extern DMA_HandleTypeDef hdma_usart1_rx;
  24. extern DMA_HandleTypeDef hdma_usart1_tx;
  25. /* Private variables ---------------------------------------------------------*/
  26. #define BUFFER_SIZE 100
  27. uint8_t rx_len=0;  //接收一幀數據的長度
  28. uint8_t recv_end_flag=0;   //一幀數據接收完成標志
  29. uint8_t rx_buffer[BUFFER_SIZE]={"123"};  //接收數據緩存
  30. uint8_t tx_buffer[BUFFER_SIZE];
  31. /* USER CODE END PV */

  32. /* Private function prototypes -----------------------------------------------*/
  33. void SystemClock_Config(void);

  34. /* USER CODE BEGIN PFP */
  35. /* Private function prototypes -----------------------------------------------*/

  36. /* USER CODE END PFP */

  37. /* USER CODE BEGIN 0 */

  38. /* USER CODE END 0 */

  39. /**
  40.   * @brief  The application entry point.
  41.   *
  42.   * @retval None
  43.   */
  44. int main(void)
  45. {
  46.   /* USER CODE BEGIN 1 */
  47.   /* USER CODE END 1 */

  48.   /* MCU Configuration----------------------------------------------------------*/

  49.   /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  50.   HAL_Init();

  51.   /* USER CODE BEGIN Init */

  52.   /* USER CODE END Init */

  53.   /* Configure the system clock */
  54.   SystemClock_Config();

  55.   /* USER CODE BEGIN SysInit */

  56.   /* USER CODE END SysInit */

  57.   /* Initialize all configured peripherals */
  58.   MX_GPIO_Init();
  59.   MX_DMA_Init();
  60.   MX_TIM2_Init();
  61.   MX_USART1_UART_Init();
  62.   /* USER CODE BEGIN 2 */
  63.   //printf("tst uart DMA begin\r\n");
  64.   //__HAL_UART_ENABLE_IT(&huart1, UART_IT_RXNE);
  65.         __HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE);                         //使能IDLE中斷
  66.         HAL_UART_Receive_DMA(&huart1,rx_buffer,BUFFER_SIZE);//啟動DMA接收
  67.   /* USER CODE END 2 */

  68.   /* Infinite loop */
  69.   /* USER CODE BEGIN WHILE */
  70.   while (1)
  71.   {
  72.           HAL_GPIO_WritePin(Tst_Pin_GPIO_Port, Tst_Pin_Pin, GPIO_PIN_RESET);
  73.          
  74.                 if(recv_end_flag ==1)
  75.                 {
  76.                         HAL_UART_Transmit(&huart1,rx_buffer, rx_len,200);//接收數據打印出來
  77.                         rx_len=0;
  78.                         recv_end_flag=0;
  79.                         //memset(rx_buffer,0x00,BUFFER_SIZE);
  80.                         HAL_UART_Receive_DMA(&huart1,rx_buffer,BUFFER_SIZE);                //重啟DMA接收
  81.                 }       
  82.                 HAL_GPIO_WritePin(Tst_Pin_GPIO_Port, Tst_Pin_Pin, GPIO_PIN_SET);
  83.   /* USER CODE END WHILE */

  84.   /* USER CODE BEGIN 3 */
  85.   
  86.   
  87.   /* USER CODE END 3 */

  88.         }
  89. }

  90. /**
  91.   * @brief System Clock Configuration
  92.   * @retval None
  93.   */
  94. void SystemClock_Config(void)
  95. {

  96.   RCC_OscInitTypeDef RCC_OscInitStruct;
  97.   RCC_ClkInitTypeDef RCC_ClkInitStruct;

  98.     /**Initializes the CPU, AHB and APB busses clocks
  99.     */
  100.   RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI;
  101.   RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  102.   RCC_OscInitStruct.HSICalibrationValue = 16;
  103.   RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
  104.   if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  105.   {
  106.     _Error_Handler(__FILE__, __LINE__);
  107.   }

  108.     /**Initializes the CPU, AHB and APB busses clocks
  109.     */
  110.   RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
  111.                               |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  112.   RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI;
  113.   RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  114.   RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
  115.   RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  116.   if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK)
  117.   {
  118.     _Error_Handler(__FILE__, __LINE__);
  119.   }

  120.   HAL_RCC_MCOConfig(RCC_MCO, RCC_MCO1SOURCE_SYSCLK, RCC_MCODIV_1);

  121.     /**Configure the Systick interrupt time
  122.     */
  123.   HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq()/1000);

  124.     /**Configure the Systick
  125.     */
  126.   HAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK);

  127.   /* SysTick_IRQn interrupt configuration */
  128.   HAL_NVIC_SetPriority(SysTick_IRQn, 0, 0);
  129. }

  130. /* USER CODE BEGIN 4 */
  131. void UART_IDLE_Callback(UART_HandleTypeDef *huart)
  132. {
  133.         uint16_t i = 0;
  134.   if((__HAL_UART_GET_FLAG(huart,UART_FLAG_IDLE) != RESET))  
  135.   {
  136.                 if(huart->Instance == USART1)
  137.                 {
  138.                         __HAL_UART_CLEAR_IDLEFLAG(huart);
  139.                         i = huart->Instance->SR;
  140.                         i = huart->Instance->DR;
  141.                         i = hdma_usart1_rx.Instance->CNDTR;         //讀取DMA剩余傳輸數量
  142.                         HAL_UART_DMAStop(huart);
  143.                        
  144.                         /* 此處處理數據,主要是拷貝和置位標志位 */
  145.                         if((recv_end_flag == 0)&&(i!=BUFFER_SIZE))
  146.                         {
  147.                                 rx_len = BUFFER_SIZE - i;
  148.                                 //memcpy(tx_buffer,rx_buffer,rx_len);
  149.                                 recv_end_flag = 1;
  150.                         }
  151.                         else
  152.                         {
  153.                                 /* 清空緩存,重新接收 */
  154.                                 //memset(rx_buffer,0x00,BUFFER_SIZE);
  155.                                 HAL_UART_Receive_DMA(huart,(uint8_t *)&rx_buffer,BUFFER_SIZE);
  156.                         }
  157.                 }
  158.   }
  159.   
  160. }  
  161. /* USER CODE END 4 */

  162. /**
  163.   * @brief  This function is executed in case of error occurrence.
  164.   * @param  file: The file name as string.
  165.   * @param  line: The line in file as a number.
  166.   * @retval None
  167.   */
  168. void _Error_Handler(char *file, int line)
  169. {
  170.   /* USER CODE BEGIN Error_Handler_Debug */
  171.   /* User can add his own implementation to report the HAL error return state */
  172.   while(1)
  173.   {
  174.   }
  175.   /* USER CODE END Error_Handler_Debug */
  176. }

  177. #ifdef  USE_FULL_ASSERT
  178. /**
  179.   * @brief  Reports the name of the source file and the source line number
  180.   *         where the assert_param error has occurred.
  181.   * @param  file: pointer to the source file name
  182.   * @param  line: assert_param error line source number
  183.   * @retval None
  184.   */
  185. void assert_failed(uint8_t* file, uint32_t line)
  186. {
  187.   /* USER CODE BEGIN 6 */
  188.   /* User can add his own implementation to report the file name and line number,
  189.     ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  190.   /* USER CODE END 6 */
  191. }
  192. #endif /* USE_FULL_ASSERT */

  193. /**
  194.   * @}
  195.   */

  196. /**
  197.   * @}
  198.   */

  199. /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/
復制代碼

所有資料51hei提供下載:
103_UARTDMA.rar (445.2 KB, 下載次數: 104)

評分

參與人數 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 国内精品久久久久 | 国产精品99久久久久久宅男 | 精品影院| 国产在线一区二区三区 | 欧美日韩一区二区在线观看 | 亚洲激情在线观看 | 黄色毛片免费看 | 欧美成人精品一区二区男人看 | 日本超碰在线 | 久久精品亚洲精品国产欧美 | 久久久精品综合 | 国产女人精品视频 | 欧美aa在线 | 人人草人人干 | 久久免费视频网 | 日韩成人在线播放 | 久久极品| 亚洲精品一区二区 | 美女在线一区二区 | 一区二区三区日韩 | 天天操精品视频 | 91夜色在线观看 | 美国黄色毛片 | 日韩一区二区三区视频在线观看 | 日韩国产一区二区三区 | 国产精品欧美一区二区三区 | 亚洲精品久久 | wwww.8888久久爱站网 | 麻豆久久久久久 | 一区二区三区免费观看 | 美女久久久久久久久 | 韩日视频在线观看 | 91视频在线| 国产玖玖 | 久一精品 | www.99热.com| 久久国产精品网站 | 色接久久| 国产亚洲一区二区三区在线观看 | 看黄在线| 国产精品精品视频一区二区三区 |