STM32CubeMX系列|内部温度传感器

内部温度传感器

1. 内部温度传感器简介

STM32F1有一个内部的温度传感器,可以用来测量CPU及周围的温度。该温度传感器在内部和ADCx_IN16输入通道相连接,此通道把传感器输出的电压转换成数字值,温度传感器模拟输入推荐采样时间是17.1us,内部温度传感器支持的温度范围为:-40~125℃,精度为±1.5℃左右

在这里插入图片描述
通过读取ADC通道16的值,通过如下计算公式即可算出当前温度:

T = { (V25 - Vsense) / Avg_Slope } + 25

上式中:

V25 = Vsense在25℃时的数值(典型值为:1.43)
Avg_Slope = 温度与Vsense曲线的平均斜率(典型值为:4.3 mV/℃)

以上典型值请参考数据手册的电气特性章节中的介绍

在这里插入图片描述

2. 硬件设计

本实验通过ADC1通道16采集内部温度,将采样的AD值和转换后的温度值通过USART1串口打印出来,同时D1指示灯闪烁,提示系统正常运行

  • D1指示灯
  • ADC1_INT1
  • USART1串口
3. 软件设计
3.1 STM32CubeMX设置
  • RCC设置外接HSE,时钟设置为72M,ADC预分频因子设置为6,ADC_CLK为12MHz
  • PC0设置为GPIO推挽输出模式、上拉、高速、默认输出电平为高电平
  • USART1选择为异步通讯方式,波特率设置为115200Bits/s,传输数据长度为8Bit,无奇偶校验,1位停止位
  • 激活ADC1温度传感器通道,设置右对齐,关闭扫描、连续及间断模式,使能regular conversion,设置软件触发、设置采样时间239.5个周期(19.96us)

在这里插入图片描述

  • 输入工程名,选择工程路径(不要有中文),选择MDK-ARM V5;勾选Generated periphera initialization as a pair of ‘.c/.h’ files per IP ;点击GENERATE CODE,生成工程代码
3.2 MDK-ARM编程
  • 在adc.c文件中可以看到ADC初始化函数
void MX_ADC1_Init(void){
  ADC_ChannelConfTypeDef sConfig = {0};
  /** Common config */
  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();
  }
  /** Configure Regular Channel*/
  sConfig.Channel = ADC_CHANNEL_TEMPSENSOR;
  sConfig.Rank = ADC_REGULAR_RANK_1;
  sConfig.SamplingTime = ADC_SAMPLETIME_239CYCLES_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK){
    Error_Handler();
  }
}

void HAL_ADC_MspInit(ADC_HandleTypeDef* adcHandle){
  if(adcHandle->Instance==ADC1)  {
    /* ADC1 clock enable */
    __HAL_RCC_ADC1_CLK_ENABLE();
  }
}
  • 在主函数while循环中添加如下测试程序
while (1){
	HAL_ADC_Start(&hadc1);	//启动ADC转换
	HAL_ADC_PollForConversion(&hadc1,10);	//等待转换完成,10ms表示超时时间
	AD_Value = HAL_ADC_GetValue(&hadc1);	//读取ADC转换数据(12位数据)
	printf("ADC1_IN16 ADC value: %d\r\n",AD_Value);
	Vol_Value = AD_Value*(3.3/4096);	//AD值乘以分辨率即为电压值
	printf("ADC1_IN16 VOL value: %.2fV\r\n",Vol_Value);
	Temperature = (1.43 - Vol_Value)/0.0043 + 25;	//根据公式算出温度值
	printf("MCU Internal Temperature: %.2f¡æ\r\n",Temperature);
	printf("\r\n");
  
	HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_0);
	HAL_Delay(1000);
}
4. 下载验证

编译无误后下载到开发板,可以看到系统运行时D1指示灯不断闪烁,串口不断的打印读取的温度数据
在这里插入图片描述

关注我的公众号,在公众号里发如下消息,即可获取相应的工程源代码:

玩转STM32CubeMX | 内部温度传感器

在这里插入图片描述

  • 7
    点赞
  • 68
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 7
    评论
以下是使用STM32F103C6T6内部温度传感器的代码: 首先,需要在CubeMX中配置ADC模块以读取内部温度传感器的值。选择对应的引脚和通道,并设置ADC时钟和采样时间。然后在"Analog Watchdog"选项卡中,启用内部温度传感器并选择对应的通道。 在main.c文件中,包含以下头文件: ```c #include "stm32f1xx_hal.h" #include <stdio.h> ``` 然后定义一些变量和函数: ```c ADC_HandleTypeDef hadc1; float temperature; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_ADC1_Init(void); void HAL_ADC_MspInit(ADC_HandleTypeDef* hadc); void Error_Handler(void); ``` 在main函数中,调用函数初始化系统时钟和GPIO: ```c HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); ``` 然后初始化ADC模块: ```c MX_ADC1_Init(); ``` 在while循环中,读取内部温度传感器的值,并计算温度: ```c HAL_ADC_Start(&hadc1); //启动ADC转换 HAL_ADC_PollForConversion(&hadc1, 100); //等待转换完成 uint16_t adc_val = HAL_ADC_GetValue(&hadc1); //获取转换结果 temperature = ((1.43 - (adc_val * 3.3 / 4096)) / 0.0043) + 25; //计算温度 ``` 完整代码如下: ```c #include "stm32f1xx_hal.h" #include <stdio.h> ADC_HandleTypeDef hadc1; float temperature; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_ADC1_Init(void); void HAL_ADC_MspInit(ADC_HandleTypeDef* hadc); void Error_Handler(void); int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_ADC1_Init(); while (1) { HAL_ADC_Start(&hadc1); //启动ADC转换 HAL_ADC_PollForConversion(&hadc1, 100); //等待转换完成 uint16_t adc_val = HAL_ADC_GetValue(&hadc1); //获取转换结果 temperature = ((1.43 - (adc_val * 3.3 / 4096)) / 0.0043) + 25; //计算温度 } } void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; /** Initializes the RCC Oscillators according to the specified parameters * in the RCC_OscInitTypeDef structure. */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; RCC_OscInitStruct.HSEState = RCC_HSE_ON; RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1; 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(); } /** Initializes the CPU, AHB and APB buses clocks */ 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(); } } static void MX_ADC1_Init(void) { ADC_ChannelConfTypeDef sConfig = {0}; hadc1.Instance = ADC1; hadc1.Init.ScanConvMode = 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(); } sConfig.Channel = ADC_CHANNEL_TEMPSENSOR; sConfig.Rank = 1; sConfig.SamplingTime = ADC_SAMPLETIME_239CYCLES_5; if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK) { Error_Handler(); } } static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; __HAL_RCC_GPIOC_CLK_ENABLE(); HAL_GPIO_WritePin(GPIOC, GPIO_PIN_13, GPIO_PIN_RESET); GPIO_InitStruct.Pin = GPIO_PIN_13; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOC, &GPIO_InitStruct); } void HAL_ADC_MspInit(ADC_HandleTypeDef* hadc) { GPIO_InitTypeDef GPIO_InitStruct = {0}; if(hadc->Instance==ADC1) { /* USER CODE BEGIN ADC1_MspInit 0 */ /* USER CODE END ADC1_MspInit 0 */ /* Peripheral clock enable */ __HAL_RCC_ADC1_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); /**ADC1 GPIO Configuration PA1 ------> ADC1_IN1 */ GPIO_InitStruct.Pin = GPIO_PIN_1; GPIO_InitStruct.Mode = GPIO_MODE_ANALOG; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); /* USER CODE BEGIN ADC1_MspInit 1 */ /* USER CODE END ADC1_MspInit 1 */ } } void Error_Handler(void) { __disable_irq(); while (1) { } } ``` 需要注意的是,内部温度传感器的精度相对较低,可能存在较大的误差。因此,建议使用外部温度传感器来获取更准确的温度值。
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

安迪西嵌入式

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值