VL53L8CX TOF开发(3)----检测阈值

描述

概述

最近在弄ST的课程,需要样片的可以加群申请:615061293 。

本章展示如何使用VL53L8CX近接传感器的"检测阈值"功能。这个功能允许用户为传感器设置预定义的条件,当这些条件满足时,传感器可以触发一个中断。

stm32cubemx


VL53L8CX传感器允许用户更灵活地定义响应行为,特别是当检测到特定的测量结果时。例如,可以设置当对象的距离低于或高于特定值时,触发中断。这种功能在各种实际应用中,如智能开关、安全系统或机器人导航中,都非常有用。

视频教学

https://www.bilibili.com/video/BV1Qf421R71U/

样品申请

https://www.wjx.top/vm/OhcKxJk.aspx#

stm32cubemx

源码下载

https://download.csdn.net/download/qq_24312945/89328194

实现demo

配置VL53L8CX传感器以实现特定条件下的目标检测和距离测量。通过设置信号强度和距离阈值,可以根据具体应用需求定制传感器的检测行为,使其在各种场景中发挥作用。
实现为每个区域(在4x4分辨率中有16个区域)设定了两个阈值:一个基于信号强度,另一个基于物体的测量距离。

硬件准备

首先需要准备一个开发板,这里我准备的是自己绘制的开发板,需要的可以进行申请。
主控为STM32G431CB,TOF为VL53L8CX
 

stm32cubemx

生成STM32CUBEMX

选择MCU

测试版所用的MCU为STM32G431CB。
 

stm32cubemx

串口配置

查看原理图,PA9和PA10设置为开发板的串口。
 

stm32cubemx

配置串口。
 

stm32cubemx

IIC配置

在这个应用中,VL53L8CX模块通过I2C(IIC)接口与主控器通信。具体来说,VL53L8CX模块的I2C引脚连接到主控器的PA8和PB5两个IO口。

stm32cubemxstm32cubemx

配置IIC为快速模式,速度为400k。

stm32cubemx

LPn 设置

若进行IIC通讯,LPn设置为高电平状态。

stm32cubemx

这里对应管脚为PA12。

stm32cubemx

配置为PA12。

stm32cubemx

INT设置

自主模式可以通过获取INT管脚进行判断数据是否准备好。

stm32cubemx


配置PA11为输入模式。
 

stm32cubemx

X-CUBE-TOF1

本节介绍在不需要使用样例应用时如何使用STM32CubeMX将X-CUBE-TOF1软件包添加到项目中。有了这样的设置,就只配置了驱动层。
 

stm32cubemx

串口重定向

打开魔术棒,勾选MicroLIB
 

在main.c中,添加头文件,若不添加会出现 identifier “FILE” is undefined报错。

 

/* USER CODE BEGIN Includes */
#include "stdio.h"
/* USER CODE END Includes */

 

函数声明和串口重定向:

 

/* USER CODE BEGIN PFP */
int fputc(int ch, FILE *f){
	HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
	return ch;
}
/* USER CODE END PFP */

 

代码配置

在custom_ranging_sensor.c代码中,有IO口驱动VL53L8CX进行复位的代码,由于没有配置对应的IO,所以需要注释掉。

stm32cubemx

TOF代码配置

在main.c中添加对应头文件。

 

/* USER CODE BEGIN Includes */
#include "stdio.h"
#include "vl53l8cx.h"
#include "custom_ranging_sensor.h"

/* USER CODE END Includes */

 

函数与变量定义:

 

/* USER CODE BEGIN PFP */
int fputc(int ch, FILE *f){
	HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
	return ch;
}


static int32_t status = 0;
static RANGING_SENSOR_Result_t Result;

/* USER CODE END PFP */

 

kcps阈值检测

在激光雷达和飞行时间(ToF)传感器的测量中,kcps 代表 每秒千计数(kilo counts per second)。这是一个用于表示传感器每秒钟探测到的光子返回计数的单位。具体来说,它反映了每秒钟每个感光单元(SPAD)检测到的光子数量。
较高的kcps值通常表示反射率较高的目标或较近的目标,而较低的kcps值则表示反射率较低的目标或较远的目标。

 

	/*********************************/
	/*  程序检测阈值 */
	/*********************************/

    /* 在此示例中,我们希望每个区域有2个阈值,分辨率为4x4 */
    /* 创建阈值数组(大小不能更改) */
	VL53L8CX_DetectionThresholds thresholds[VL53L8CX_NB_THRESHOLDS];	
	
  /* Set all values to 0 */
  memset(&thresholds, 0, sizeof(thresholds));

	
  VL53L8CX_Object_t *pL8obj = CUSTOM_RANGING_CompObj[CUSTOM_VL53L8CX];

    /* 为所有区域添加阈值(在4x4的分辨率中有16个区域,或在8x8中有64个) */
  for (int i = 0; i < 16; i++) {
		/* 第一个所需的阈值是GREATER_THAN模式。请注意,第一个必须始终使用数学操作VL53L8CX_OPERATION_NONE设置。
		 * 在此示例中,信号阈值设置为150 kcps/spads(格式会在驱动程序内自动更新)
		 */
		
    thresholds[i].zone_num = i;
    thresholds[i].measurement = VL53L8CX_SIGNAL_PER_SPAD_KCPS;
    thresholds[i].type = VL53L8CX_IN_WINDOW;
    thresholds[i].mathematic_operation = VL53L8CX_OPERATION_NONE;
    thresholds[i].param_low_thresh = 1500;
    thresholds[i].param_high_thresh = 1700;		
  }

	/* 必须明确指出最后的阈值。因为我们有16个检查器(16个区域x 1),所以最后一个是第15个 */
  thresholds[15].zone_num = VL53L8CX_LAST_THRESHOLD | thresholds[15].zone_num;

	/* 向传感器发送阈值数组 */
	status |= vl53l8cx_set_detection_thresholds(&pL8obj- >Dev, thresholds);

	/* 启用阈值检测 */
	status |= vl53l8cx_set_detection_thresholds_enable(&pL8obj- >Dev, 1U);

	/* 设置传感器的测量频率,这决定了传感器执行测量的速度 */
	status |= vl53l8cx_set_ranging_frequency_hz(&(pL8obj- >Dev), 10);
	
	
  if (status != VL53L8CX_STATUS_OK)
  {
    printf("ERROR : Configuration programming error!nn");
    while (1);
  }

  status = vl53l8cx_start_ranging(&(pL8obj- >Dev));
  if (status != VL53L8CX_STATUS_OK)
  {
    printf("vl53l5cx_start_ranging failedn");
    while (1);
  }



  static VL53L8CX_ResultsData data;

 

检测类型

这些宏定义用于在配置VL53L8CX传感器时指定“检测器”的类型,它们表示测量值的窗口或范围,由低阈值和高阈值定义。每个宏定义代表不同的检测条件类型。

 

/**
 * @brief The following macro are used to define the 'type' of a checker.
 * They indicate the window of measurements, defined by low and a high
 * thresholds.
 */

#define VL53L8CX_IN_WINDOW					((uint8_t)0U)
#define VL53L8CX_OUT_OF_WINDOW				((uint8_t)1U)
#define VL53L8CX_LESS_THAN_EQUAL_MIN_CHECKER ((uint8_t)2U)
#define VL53L8CX_GREATER_THAN_MAX_CHECKER	((uint8_t)3U)
#define VL53L8CX_EQUAL_MIN_CHECKER			((uint8_t)4U)
#define VL53L8CX_NOT_EQUAL_MIN_CHECKER		((uint8_t)5U)

 

VL53L8CX_IN_WINDOW:如果测量值在指定的低阈值和高阈值之间,则检测为真。
VL53L8CX_OUT_OF_WINDOW:如果测量值在指定的低阈值和高阈值之外,则检测为真。
VL53L8CX_LESS_THAN_EQUAL_MIN_CHECKER: 如果测量值小于或等于指定的低阈值,则检测为真。
VL53L8CX_GREATER_THAN_MAX_CHECKER:如果测量值大于指定的高阈值,则检测为真。
VL53L8CX_EQUAL_MIN_CHECKER:如果测量值等于指定的低阈值,则检测为真。
VL53L8CX_NOT_EQUAL_MIN_CHECKER:如果测量值不等于指定的低阈值,则检测为真。

距离阈值检测

配置VL53L8CX传感器进行距离测量,并设置每个区域的检测阈值,使传感器能够检测特定距离范围内的目标。通过设置测量频率和启动测量,传感器将开始定期进行测距,并根据配置的阈值进行检测。这些步骤确保传感器按预期工作,并在指定条件下触发相应的检测结果。

 

	/*********************************/
	/*  程序检测阈值 */
	/*********************************/

    /* 在此示例中,我们希望每个区域有2个阈值,分辨率为4x4 */
    /* 创建阈值数组(大小不能更改) */
	VL53L8CX_DetectionThresholds thresholds[VL53L8CX_NB_THRESHOLDS];	
	
  /* Set all values to 0 */
  memset(&thresholds, 0, sizeof(thresholds));

	
  VL53L8CX_Object_t *pL8obj = CUSTOM_RANGING_CompObj[CUSTOM_VL53L8CX];


    /* 为所有区域添加阈值(在4x4的分辨率中有16个区域,或在8x8中有64个) */
  for (int i = 0; i < 16; i++) {	
		/* 第二个所需的检查器是IN_WINDOW模式。我们将设置一个数学阈值VL53L5CX_OPERATION_OR,以将前一个检查器添加到此检查器。
		 * 在此示例中,距离阈值设置在200mm和400mm之间(格式会在驱动程序内自动更新)。
		 */		
    thresholds[i].zone_num = i;
    thresholds[i].measurement = VL53L8CX_DISTANCE_MM;
    thresholds[i].type = VL53L8CX_IN_WINDOW;
    thresholds[i].mathematic_operation = VL53L8CX_OPERATION_OR;
    thresholds[i].param_low_thresh = 200;
    thresholds[i].param_high_thresh = 400;		
		
	}
	

	/* 必须明确指出最后的阈值。因为我们有16个检查器(16个区域x 1),所以最后一个是第15个 */
  thresholds[15].zone_num = VL53L8CX_LAST_THRESHOLD | thresholds[15].zone_num;

	/* 向传感器发送阈值数组 */
	status |= vl53l8cx_set_detection_thresholds(&pL8obj- >Dev, thresholds);

	/* 启用阈值检测 */
	status |= vl53l8cx_set_detection_thresholds_enable(&pL8obj- >Dev, 1U);

	/* 设置传感器的测量频率,这决定了传感器执行测量的速度 */
	status |= vl53l8cx_set_ranging_frequency_hz(&(pL8obj- >Dev), 10);
	
	
  if (status != VL53L8CX_STATUS_OK)
  {
    printf("ERROR : Configuration programming error!nn");
    while (1);
  }

  status = vl53l8cx_start_ranging(&(pL8obj- >Dev));
  if (status != VL53L8CX_STATUS_OK)
  {
    printf("vl53l5cx_start_ranging failedn");
    while (1);
  }



  static VL53L8CX_ResultsData data;

 

主程序

主程序来获取对应的INT位状态来判定数据是否准备好。

 

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
	if(HAL_GPIO_ReadPin  ( GPIOA, GPIO_PIN_11) ==0)		
	{
	
		// 获取传感器的测距数据
		status = vl53l8cx_get_ranging_data(&(pL8obj- >Dev), &data);
		printf("n");
		// 循环打印所有16个区域的数据
		for (int i = 0; i < 16; i++) {
			printf("Zone : %3d, Status : %3u, Distance : %4d mm, Signal : %5lu kcps/SPADsrn",
							 i,
							 data.target_status[VL53L8CX_NB_TARGET_PER_ZONE * i],
							 data.distance_mm[VL53L8CX_NB_TARGET_PER_ZONE * i],
							 data.signal_per_spad[VL53L8CX_NB_TARGET_PER_ZONE * i]);
		}	
	}
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */

 

演示结果

1500-1700kcps/SPADs检测。

stm32cubemx

200-400mm距离检测。

stm32cubemx


审核编辑 黄宇

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分