要在 STM32 上使用 HAL 库读取 ADC 值并通过串口传输数据,可以按照以下步骤进行。本文将以 STM32F1 系列为例,详细介绍如何配置 ADC 和 UART,并实现数据的读取与传输。
注意:确保电压匹配,STM32 的逻辑电平通常为 3.3V,避免直接连接 5V 设备。
在生成的代码基础上,编写主程序以实现以下功能:
main.c
中,确保 HAL 库已正确初始化 ADC 和 UART。以下是一个完整的 main.c
示例,展示如何读取 ADC 值并通过 UART 发送:
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include <stdio.h>
/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_ADC1_Init(void);
static void MX_USART1_UART_Init(void);
/* Private variables ---------------------------------------------------------*/
ADC_HandleTypeDef hadc1;
UART_HandleTypeDef huart1;
/* Function prototypes -------------------------------------------------------*/
#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__ */
/* Private user code ---------------------------------------------------------*/
/**
* @brief The application entry point.
* @retval int
*/
int main(void)
{
/* MCU 初始化 */
HAL_Init();
/* 配置系统时钟 */
SystemClock_Config();
/* 初始化所有配置的外设 */
MX_GPIO_Init();
MX_ADC1_Init();
MX_USART1_UART_Init();
/* 配置 printf 通过 UART 输出 */
__HAL_UART_ENABLE(&huart1);
char msg[50];
uint32_t adc_value;
float voltage;
/* 无限循环 */
while (1)
{
/* 启动 ADC 转换 */
if (HAL_ADC_Start(&hadc1) == HAL_OK)
{
/* 等待转换完成 */
if (HAL_ADC_PollForConversion(&hadc1, HAL_MAX_DELAY) == HAL_OK)
{
/* 获取 ADC 值 */
adc_value = HAL_ADC_GetValue(&hadc1);
/* 将 ADC 值转换为电压(假设 VREF=3.3V,12位分辨率) */
voltage = (adc_value * 3.3f) / 4095.0f;
/* 格式化字符串 */
snprintf(msg, sizeof(msg), "ADC Value: %lu, Voltage: %.2f V\r\n", adc_value, voltage);
/* 通过 UART 发送 */
HAL_UART_Transmit(&huart1, (uint8_t*)msg, strlen(msg), HAL_MAX_DELAY);
}
}
/* 延时(例如 500ms) */
HAL_Delay(500);
}
}
/**
* @brief System Clock Configuration
* @retval None
*/
void SystemClock_Config(void)
{
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
/** 初始化 RCC 振荡器
*/
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
RCC_OscInitStruct.HSEState = RCC_HSE_ON;
RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
RCC_OscInitStruct.HSIState = RCC_HSI_ON;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
{
Error_Handler();
}
/** 初始化 RCC 时钟
*/
RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
{
Error_Handler();
}
}
/**
* @brief ADC1 Initialization Function
* @param None
* @retval None
*/
static void MX_ADC1_Init(void)
{
/* 用户可以在此处调整 ADC 参数 */
ADC_ChannelConfTypeDef sConfig = {0};
/** 初始化 ADC
*/
hadc1.Instance = ADC1;
hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
hadc1.Init.ContinuousConvMode = DISABLE;
hadc1.Init.DiscontinuousConvMode = DISABLE;
hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
hadc1.Init.NbrOfConversion = 1;
if (HAL_ADC_Init(&hadc1) != HAL_OK)
{
Error_Handler();
}
/** 配置 ADC 通道
*/
sConfig.Channel = ADC_CHANNEL_0; // PA0
sConfig.Rank = 1;
sConfig.SamplingTime = ADC_SAMPLETIME_239CYCLES_5;
if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
{
Error_Handler();
}
}
/**
* @brief USART1 Initialization Function
* @param None
* @retval None
*/
static void MX_USART1_UART_Init(void)
{
/* 用户可以在此处调整 UART 参数 */
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_TX_RX;
huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;
huart1.Init.OverSampling = UART_OVERSAMPLING_16;
if (HAL_UART_Init(&huart1) != HAL_OK)
{
Error_Handler();
}
}
/**
* @brief GPIO Initialization Function
* @param None
* @retval None
*/
static void MX_GPIO_Init(void)
{
__HAL_RCC_GPIOA_CLK_ENABLE();
}
/**
* @brief Retargets the C library printf function to the UART.
* @param None
* @retval None
*/
PUTCHAR_PROTOTYPE
{
/* 将字符通过 UART 发送 */
HAL_UART_Transmit(&huart1, (uint8_t *)&ch, 1, HAL_MAX_DELAY);
return ch;
}
/**
* @brief This function is executed in case of error occurrence.
* @retval None
*/
void Error_Handler(void)
{
/* 用户可以添加自己的错误处理代码,例如 LED 指示 */
while(1)
{
}
}
printf
函数到 UART,以便使用 printf
发送数据。printf
将 ADC 值和电压通过 UART 发送。可能原因:
解决方法:
MX_USART1_UART_Init
函数中的波特率是否与串口终端设置一致。可能原因:
解决方法:
MX_ADC1_Init
函数中 ADC 通道是否配置为模拟输入(在 STM32CubeMX 中设置引脚为 "Analog")。ADC_SAMPLETIME_71CYCLES_5
或更长)。可能原因:
解决方法:
ADC_SAMPLETIME_239CYCLES_5
)以提高精度。HAL_ADCEx_Calibration_Start
)。如果需要读取多个 ADC 通道,可以配置扫描模式或多通道转换,并在代码中处理多个值。
领取专属 10元无门槛券
手把手带您无忧上云