RoboMaster

标题: 【分享帖】从STM32开始的RoboMaster生活:进阶篇 I [GPIO] [打印本页]

作者: Alchemic Ronin    时间: 2020-3-12 11:00
标题: 【分享帖】从STM32开始的RoboMaster生活:进阶篇 I [GPIO]
本文已经同步发布于作者部署的私人博客
为了更好的排版和观看体验
可以移步到 从STM32开始的RoboMaster生活:进阶篇 I [GPIO]

从STM32开始的RoboMaster生活:进阶篇 I [GPIO]
0.0 一点点声明1.0 什么是GPIO?
GPIO接脚可以供使用者由程控自由使用,PIN脚依现实考量可作为通用输入 ( GPI ) 或通用输出 ( GPO ) 或通用输入与输出 ( GPIO ) 。在嵌入式系统中,经常需要控制许多结构简单的外部设备或者电路,这些设备有的需要通过CPU控制,有的需要CPU提供输入信号。对设备的控制,使用传统的串口或者并口就显得比较复杂,所以,在嵌入式微控制器上通常提供了一种“通用可编程I/O端口”,也就是GPIO。一个GPIO端口至少需要两个寄存器,一个做控制用的“通用IO端口控制寄存器”,还有一个是存放数据的“通用I/O端口数据寄存器”。数据寄存器的每一位是和GPIO的硬件引脚对应的,而数据的传递方向是通过控制寄存器设置的,通过控制寄存器可以设置每一位引脚的数据流向。
2.0 GPIO在哪里?

                               
登录/注册后可看帖子
​        上图中,26即为GPIO,位于开发板A型的中间位置

                               
登录/注册后可看帖子
​        这是GPIO的具体引脚图,在配置芯片引脚时请务必核实引脚号是否正确

                               
登录/注册后可看帖子

                               
登录/注册后可看帖子
3.0 如何使用GPIO?
在下文中,几乎任何HAL函数的参数都需要外设和引脚,但是在比如Arduino的平台上,只需要引脚即可。其原因在于,对于STM32,一个外设可以连接众多引脚,在配置的时候,这些引脚都使用同一份外设配置,但是实际使用的时候,可以分别控制。

3.1 读取GPIO
  1. GPIO_PinState HAL_GPIO_ReadPin(GPIO_TypeDef *GPIOx, uint16_t GPIO_Pin)
复制代码
3.2 写入GPIO
  1. void HAL_GPIO_WritePin(GPIO_TypeDef *GPIOx, uint16_t GPIO_Pin, GPIO_PinState PinState)
复制代码
3.3 反转GPIO
  1. void HAL_GPIO_TogglePin(GPIO_TypeDef *GPIOx, uint16_t GPIO_Pin)
复制代码
如果GPIO状态为GPIO_PIN_RESET,则改为GPIO_PIN_SET;如果GPIO状态为GPIO_PIN_SET,则改为GPIO_PIN_RESET。

3.4 锁死GPIO
  1. HAL_StatusTypeDef HAL_GPIO_LockPin(GPIO_TypeDef *GPIOx, uint16_t GPIO_Pin)
复制代码
锁死的是该GPIO的配置,而不是状态。任何尝试修改其配置的操作都会失败,除非重置GPIO。

3.5 重置GPIO
  1. void HAL_GPIO_DeInit(GPIO_TypeDef *GPIOx, uint32_t GPIO_Pin)
复制代码
重置GPIO到默认的配置(Input Floating模式)。用于当我们不再使用某个外设时,节约电量,避免额外的消耗。

4.0 练习项目
4.1 项目简介4.2 芯片配置

                               
登录/注册后可看帖子

                               
登录/注册后可看帖子
4.3 项目代码

                               
登录/注册后可看帖子
  1. /* USER CODE BEGIN Header */
  2. /**
  3.   ******************************************************************************
  4.   * @file           : main.c
  5.   * @brief          : Main program body
  6.   ******************************************************************************
  7.   * @attention
  8.   *
  9.   * <h2><center>© Copyright (c) 2020 STMicroelectronics.
  10.   * All rights reserved.</center></h2>
  11.   *
  12.   * This software component is licensed by ST under BSD 3-Clause license,
  13.   * the "License"; You may not use this file except in compliance with the
  14.   * License. You may obtain a copy of the License at:
  15.   *                        opensource.org/licenses/BSD-3-Clause
  16.   *
  17.   ******************************************************************************
  18.   */
  19. /* USER CODE END Header */

  20. /* Includes ------------------------------------------------------------------*/
  21. #include "main.h"

  22. /* Private includes ----------------------------------------------------------*/
  23. /* USER CODE BEGIN Includes */

  24. /* USER CODE END Includes */

  25. /* Private typedef -----------------------------------------------------------*/
  26. /* USER CODE BEGIN PTD */

  27. /* USER CODE END PTD */

  28. /* Private define ------------------------------------------------------------*/
  29. /* USER CODE BEGIN PD */
  30. /* USER CODE END PD */

  31. /* Private macro -------------------------------------------------------------*/
  32. /* USER CODE BEGIN PM */

  33. /* USER CODE END PM */

  34. /* Private variables ---------------------------------------------------------*/

  35. /* USER CODE BEGIN PV */

  36. /* USER CODE END PV */

  37. /* Private function prototypes -----------------------------------------------*/
  38. void SystemClock_Config(void);
  39. static void MX_GPIO_Init(void);
  40. /* USER CODE BEGIN PFP */

  41. /* USER CODE END PFP */

  42. /* Private user code ---------------------------------------------------------*/
  43. /* USER CODE BEGIN 0 */

  44. /* USER CODE END 0 */

  45. /**
  46.   * @brief  The application entry point.
  47.   * @retval int
  48.   */
  49. int main(void)
  50. {
  51.   /* USER CODE BEGIN 1 */

  52.   /* USER CODE END 1 */

  53.   /* MCU Configuration--------------------------------------------------------*/

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

  56.   /* USER CODE BEGIN Init */

  57.   /* USER CODE END Init */

  58.   /* Configure the system clock */
  59.   SystemClock_Config();

  60.   /* USER CODE BEGIN SysInit */

  61.   /* USER CODE END SysInit */

  62.   /* Initialize all configured peripherals */
  63.   MX_GPIO_Init();
  64.   /* USER CODE BEGIN 2 */

  65.   /* USER CODE END 2 */

  66.   /* Infinite loop */
  67.   /* USER CODE BEGIN WHILE */
  68.   while (1)
  69.   {
  70.           if(HAL_GPIO_ReadPin(Button_GPIO_Port,Button_Pin) == GPIO_PIN_SET){
  71.                   HAL_Delay(500);
  72.                   HAL_GPIO_TogglePin(LD_RED_GPIO_Port,LD_RED_Pin);
  73.                   HAL_Delay(100);
  74.                   HAL_GPIO_TogglePin(LD_GREEN_GPIO_Port,LD_GREEN_Pin);
  75.                   HAL_Delay(100);
  76.                   HAL_GPIO_TogglePin(LD1_GPIO_Port,LD1_Pin);
  77.                   HAL_Delay(100);
  78.                   HAL_GPIO_TogglePin(LD2_GPIO_Port,LD2_Pin);
  79.                   HAL_Delay(100);
  80.                   HAL_GPIO_TogglePin(LD3_GPIO_Port,LD3_Pin);
  81.                   HAL_Delay(100);
  82.                   HAL_GPIO_TogglePin(LD4_GPIO_Port,LD4_Pin);
  83.                   HAL_Delay(100);
  84.                   HAL_GPIO_TogglePin(LD5_GPIO_Port,LD5_Pin);
  85.                   HAL_Delay(100);
  86.                   HAL_GPIO_TogglePin(LD6_GPIO_Port,LD6_Pin);
  87.                   HAL_Delay(100);
  88.                   HAL_GPIO_TogglePin(LD7_GPIO_Port,LD7_Pin);
  89.                   HAL_Delay(100);
  90.                   HAL_GPIO_TogglePin(LD8_GPIO_Port,LD8_Pin);
  91.           }
  92.     /* USER CODE END WHILE */

  93.     /* USER CODE BEGIN 3 */
  94.   }
  95.   /* USER CODE END 3 */
  96. }

  97. /**
  98.   * @brief System Clock Configuration
  99.   * @retval None
  100.   */
  101. void SystemClock_Config(void)
  102. {
  103.   RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  104.   RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  105.   /** Configure the main internal regulator output voltage
  106.   */
  107.   __HAL_RCC_PWR_CLK_ENABLE();
  108.   __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE3);
  109.   /** Initializes the CPU, AHB and APB busses clocks
  110.   */
  111.   RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI;
  112.   RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  113.   RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT;
  114.   RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
  115.   if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  116.   {
  117.     Error_Handler();
  118.   }
  119.   /** Initializes the CPU, AHB and APB busses clocks
  120.   */
  121.   RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
  122.                               |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  123.   RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI;
  124.   RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  125.   RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
  126.   RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  127.   if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK)
  128.   {
  129.     Error_Handler();
  130.   }
  131. }

  132. /**
  133.   * @brief GPIO Initialization Function
  134.   * @param None
  135.   * @retval None
  136.   */
  137. static void MX_GPIO_Init(void)
  138. {
  139.   GPIO_InitTypeDef GPIO_InitStruct = {0};

  140.   /* GPIO Ports Clock Enable */
  141.   __HAL_RCC_GPIOG_CLK_ENABLE();
  142.   __HAL_RCC_GPIOB_CLK_ENABLE();
  143.   __HAL_RCC_GPIOE_CLK_ENABLE();
  144.   __HAL_RCC_GPIOF_CLK_ENABLE();

  145.   /*Configure GPIO pin Output Level */
  146.   HAL_GPIO_WritePin(GPIOG, LD8_Pin|LD7_Pin|LD6_Pin|LD5_Pin
  147.                           |LD4_Pin|LD3_Pin|LD2_Pin|LD1_Pin, GPIO_PIN_RESET);

  148.   /*Configure GPIO pin Output Level */
  149.   HAL_GPIO_WritePin(LD_RED_GPIO_Port, LD_RED_Pin, GPIO_PIN_RESET);

  150.   /*Configure GPIO pin Output Level */
  151.   HAL_GPIO_WritePin(LD_GREEN_GPIO_Port, LD_GREEN_Pin, GPIO_PIN_RESET);

  152.   /*Configure GPIO pins : LD8_Pin LD7_Pin LD6_Pin LD5_Pin
  153.                            LD4_Pin LD3_Pin LD2_Pin LD1_Pin */
  154.   GPIO_InitStruct.Pin = LD8_Pin|LD7_Pin|LD6_Pin|LD5_Pin
  155.                           |LD4_Pin|LD3_Pin|LD2_Pin|LD1_Pin;
  156.   GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
  157.   GPIO_InitStruct.Pull = GPIO_NOPULL;
  158.   GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
  159.   HAL_GPIO_Init(GPIOG, &GPIO_InitStruct);

  160.   /*Configure GPIO pin : Button_Pin */
  161.   GPIO_InitStruct.Pin = Button_Pin;
  162.   GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
  163.   GPIO_InitStruct.Pull = GPIO_NOPULL;
  164.   HAL_GPIO_Init(Button_GPIO_Port, &GPIO_InitStruct);

  165.   /*Configure GPIO pin : LD_RED_Pin */
  166.   GPIO_InitStruct.Pin = LD_RED_Pin;
  167.   GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
  168.   GPIO_InitStruct.Pull = GPIO_NOPULL;
  169.   GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
  170.   HAL_GPIO_Init(LD_RED_GPIO_Port, &GPIO_InitStruct);

  171.   /*Configure GPIO pin : LD_GREEN_Pin */
  172.   GPIO_InitStruct.Pin = LD_GREEN_Pin;
  173.   GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
  174.   GPIO_InitStruct.Pull = GPIO_NOPULL;
  175.   GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
  176.   HAL_GPIO_Init(LD_GREEN_GPIO_Port, &GPIO_InitStruct);

  177. }

  178. /* USER CODE BEGIN 4 */

  179. /* USER CODE END 4 */

  180. /**
  181.   * @brief  This function is executed in case of error occurrence.
  182.   * @retval None
  183.   */
  184. void Error_Handler(void)
  185. {
  186.   /* USER CODE BEGIN Error_Handler_Debug */
  187.   /* User can add his own implementation to report the HAL error return state */

  188.   /* USER CODE END Error_Handler_Debug */
  189. }

  190. #ifdef  USE_FULL_ASSERT
  191. /**
  192.   * @brief  Reports the name of the source file and the source line number
  193.   *         where the assert_param error has occurred.
  194.   * @param  file: pointer to the source file name
  195.   * @param  line: assert_param error line source number
  196.   * @retval None
  197.   */
  198. void assert_failed(uint8_t *file, uint32_t line)
  199. {
  200.   /* USER CODE BEGIN 6 */
  201.   /* User can add his own implementation to report the file name and line number,
  202.      tex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  203.   /* USER CODE END 6 */
  204. }
  205. #endif /* USE_FULL_ASSERT */

  206. /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/
复制代码

  ​4.4 效果展示
http://player.bilibili.com/player.html?aid=95347547&cid=162774978&page=

本文已经同步发布于作者部署的私人博客
为了更好的排版和观看体验
可以移步到 从STM32开始的RoboMaster生活:进阶篇 I [GPIO]

作者: 沧之澜    时间: 2020-3-13 10:57
建议使用代码模式并适当添加注释,增加可读性
还是感谢分享
作者: Alchemic Ronin    时间: 2020-3-13 23:09
沧之澜 发表于 2020-3-13 10:57
建议使用代码模式并适当添加注释,增加可读性
还是感谢分享

排版确实有问题,我有时间就改改,但是可以去我的博客看看,那边排版是肯定没问题的
作者: djiuser_ygxGG2o    时间: 2020-7-13 10:02
可以出一个用DT7/DR16大疆遥控器控制电机驱动的教程吗
作者: 0.0.011    时间: 2020-8-8 13:32
感谢楼主分享




欢迎光临 RoboMaster (https://bbs.robomaster.com/) Powered by Discuz! X3.2