文章目录
- 前言
- 什么是光敏传感器?
- 二、硬件电路设计
- 三、软件设计
-
- 1.CubeMX配置
- 2.CubeIDE代码
- 3.结果显示
- 总结
- 附录
前言
本篇文章用STM32CubeMX和STM32CubeIDE软件编程,主控芯片为STM32F407ZGT6驱动光敏传感器。传感器可以感知周围环境光的变化,从而实现类似于自动背光控制的应用。本文可以初步学习使用触发数据采集和处理的功能。
所用工具:
1、芯片: STM32F407ZGT6
2.驱动设备:光敏传感器
3、配置软件:STM32CubeMx
4、IDE: STM32CubeIDE
知识概括:
通过这篇文章学到:
光敏传感器原理
ADC采集数据
定时器设置
UART发送数据
什么是光敏传感器?
。它的敏感波长靠近可见光波长,包括红外波长和紫外波长。光传感器不仅局限于光的检测,还可以作为检测元件组成其他传感器,只要将这些非电转换为光信号,就可以检测到许多非电。
光敏二极管又称光电二极管。光敏二极管在结构上与半导体二极管相似,其管芯具有光敏特性 PN 结具有单向导电性,因此工作时需要增加反向电压。光敏二极管截止时,饱和反向泄漏电流非常小,即暗电流。饱和反向泄漏电流大大增加,形成光电流,随着射光强度的变化而变化。当光线照射 PN 结时,可以做到 PN 结中产生电子一空穴对,增加了少数载流子的密度。这些载流子在反向电压下漂移,增加反向电流。因此,电路中的电流可以来改变电路中的电流。
利用这种电流变化,我们串联一个电阻,将其转换为电压变化 ADC 读取电压值,判断外部光线的强弱。
使用本文ADC3的通道5读取光敏二极管电压的变化,以获得环境光的变化,并显示所获得的光强度TFTLCD上面。光敏传感器如图1所示。
二、硬件电路设计
光敏传感器只需要一个IO口连接到光敏二极管的正极,本文使用ADC3的通道5(PF7)收集光敏传感器的光强,其电路连接图如图2所示。 当环境光变化时,LS1 两端的电压也会随之改变,从而通过 ADC3_IN5 通道,读取光敏传感器上的电压,以获得环境光的强度。光线越强,电压越低,光线越暗,电压越高。
三、软件设计
1.CubeMX配置
(1) 时钟配置 如图3所示,4分别设置HSE(高速外部时钟)及时钟树的配置。HSE芯片将自动选择两个引脚连接外部晶振,如图3所示。LSE配置时钟树后,检查数据手册,了解本次使用的情况是的,为了确保其最终结果输出为整数,APB总线定时器时钟信号频率为50MHz,因此设置HCLK为100MHz,其配置图如图4所示。 (2) 调试接口配置 如图5所示,设置调试接口SW模式,占用两个芯片引脚。 (3) 定时器配置 为了便于计算定时器的分频系数和定时周期APB1设置为50MHz。先设置TIM3为内部时钟源,预分频率为4999,通过计算分频器输出的时钟信号频率为1万Hz,计数周期为500,因此TIM3定时器没500ms产生计数溢出。设置触发事件Update Event,也就是以UEV事件信号为TRGO信号。 (4) ADC配置 ADC选择3的输入通道IN5.参数部分设计如图6所示,其中外部触发源设置为主要对象。将外部触发源设置为TIM3 Trigger Out event,也就是定时器TIM3的TRGO信号,将触发转换跳变沿设置为上跳沿,因为TRGO是短时正脉冲信号。设置完成后会自动引出PF7引脚。
(5) UART配置 为了显示结果,将转换结果传输到计算机上,设置为异步模式,波特率为115200Bits/s,其UART配置如下图8所示。设置完成后,串口通信将自动引出两个引脚。 (6)中断设置 如图9所示,设置优先级分组为2位抢占2位次级UART和ADC优先级低于基本时钟。 (7)保存 在ProjectManager如图10、11所示,设置集成开发环境STM32CubeIDE。
2.CubeIDE代码
相关配置文件不做过多解释,以下只展示添加的代码段以及相应的用法,其他代码均不用动·。 (1) 头文件 作用:使用打印函数printf。 位置:位于/* USER CODE BEGIN Includes */沙箱内。
/* USER CODE BEGIN Includes */
#include "stdio.h"
#include "stdlib.h"
/* USER CODE END Includes */
(1) UART重定向 作用:使结果通过串口定向输出到屏幕上(通用代码)。 位置:位于/* USER CODE BEGIN 0 */沙箱内。
/* USER CODE BEGIN 0 */
//重定向
#ifdef __GNUC__
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif
PUTCHAR_PROTOTYPE
{
HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
return ch;
}
/* USER CODE END 0 */
(3) 开启ADC与TIM 作用:开启ADC3中断以及TIM3定时器。 位置:位于/* USER CODE BEGIN 2 */沙箱内。
/* USER CODE BEGIN 2 */
HAL_ADC_Start_IT(&hadc3); //开启ADC3中断
HAL_TIM_Base_Start(&htim3); //开启TIM3定时器
/* USER CODE END 2 */
(4) 写ADC中断回调函数 作用:ADC3在TIM3的TRGO信号的每个上跳沿启动一次ADC转换,就可以实现周期性ADC转换。 位置:位于/* USER CODE BEGIN 4 */沙箱内。
/* USER CODE BEGIN 4 */
uint32_t val;
uint16_t LSENS;
void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* adcHandle)
{
if(adcHandle->Instance==ADC3)
{
val=HAL_ADC_GetValue(&hadc3); //读取常规通道转换结果寄存器的数据
LSENS=100-(val/40); //计算,使得光强落在0-100之间
}
printf("Light intensity: %d\r\n",LSENS);
}
/* USER CODE END 4 */
3.结果显示
通过串口连接至上位机,显示结果如图12所示,改变不同的光照强度其显示为不同的结果,光照强度与光照值成正比(光照强度越强,光照值越高)。
总结
本篇文章只是定时器和ADC的简单应用,通过本次试验可以清楚地了解到ADC采集数据的过程。代码虽然简单但是很通用,适用于其他传感器。可以在代码上进行拓展,加入一些均值算法,得到较为平稳的值。本次设计参考开发板以及。
附录
完整代码:
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "adc.h"
#include "tim.h"
#include "usart.h"
#include "gpio.h"
/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "stdio.h"
#include "stdlib.h"
/* USER CODE END Includes */
/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */
/* USER CODE END PTD */
/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD */
/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */
/* USER CODE END PM */
/* Private variables ---------------------------------------------------------*/
/* USER CODE BEGIN PV */
/* USER CODE END PV */
/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */
/* USER CODE END PFP */
/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */
//重定向
#ifdef __GNUC__
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif
PUTCHAR_PROTOTYPE
{
HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
return ch;
}
/* USER CODE END 0 */
/** * @brief The application entry point. * @retval int */
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_ADC3_Init();
MX_TIM3_Init();
MX_USART1_UART_Init();
/* USER CODE BEGIN 2 */
HAL_ADC_Start_IT(&hadc3); //开启ADC3中断
HAL_TIM_Base_Start(&htim3); //开启TIM3定时器
/* USER CODE END 2 */
/* Infinite loop */
/* USER CODE BEGIN WHILE */
while (1)
{
/* 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 = {
0};
RCC_ClkInitTypeDef RCC_ClkInitStruct = {
0};
/** Configure the main internal regulator output voltage */
__HAL_RCC_PWR_CLK_ENABLE();
__HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1);
/** 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.PLL.PLLState = RCC_PLL_ON;
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
RCC_OscInitStruct.PLL.PLLM = 4;
RCC_OscInitStruct.PLL.PLLN = 100;
RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
RCC_OscInitStruct.PLL.PLLQ = 4;
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_DIV4;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_3) != HAL_OK)
{
Error_Handler();
}
}
/* USER CODE BEGIN 4 */
uint32_t val;
uint16_t LSENS;
void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* adcHandle)
{
if(adcHandle->Instance==ADC3)
{
val=HAL_ADC_GetValue(&hadc3); //读取常规通道转换结果寄存器的数据
LSENS=100-(val/40); //计算,使得光强落在0-100之间
}
printf("Light intensity: %d\r\n",LSENS);
}
/* USER CODE END 4 */
/** * @brief This function is executed in case of error occurrence. * @retval None */
void Error_Handler(void)
{
/* USER CODE BEGIN Error_Handler_Debug */
/* User can add his own implementation to report the HAL error return state */
__disable_irq();
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 */