S12SD紫外线传感器在TI MSPM0开发板上的ADC采集与强度等级转换实战
S12SD紫外线传感器在TI MSPM0开发板上的ADC采集与强度等级转换实战最近在做一个户外环境监测的小项目需要检测紫外线强度于是找到了S12SD这款紫外线传感器模块。它体积小巧价格也便宜正好搭配手头的TI MSPM0开发板来用。今天我就把整个从硬件连接到软件配置再到数据采集和强度判定的完整过程分享出来希望能帮到正在做类似项目的朋友。这篇文章会手把手带你完成S12SD传感器与MSPM0开发板的集成核心是利用开发板上的ADC模数转换器读取传感器输出的模拟电压再根据厂家提供的对照表把这个电压值转换成0到11级的紫外线强度。整个过程涉及硬件引脚选择、SysConfig图形化配置、ADC驱动编写以及数据转换逻辑我会把每个步骤都讲清楚特别是容易踩坑的地方。1. 认识S12SD紫外线传感器模块在动手接线和写代码之前咱们先花几分钟了解一下要用的传感器模块。这能帮你更好地理解后续的配置和数据含义。S12SD模块的核心是一个采用氮化镓GaN基材料的肖特基光电二极管。这种材料有啥好处呢主要是响应度高对紫外线敏感、暗电流低不受光照时的干扰信号小。传感器本身输出的电信号非常微弱所以模块上集成了一个LM358运算放大器把这个小信号放大到单片机ADC可以方便采集的电压范围。模块的主要参数如下接线和供电时要注意参数规格工作电压2.7V - 5V工作电流约 1mA测量角度130度温度漂移0.08% / °C检测波长范围240nm ~ 370nm (覆盖UVA和UVB)输出方式模拟电压输出 (ADC)接口3 Pin (VCC, GND, SIG)模块的使用很简单就三根线VCC接2.7V到5V的电源咱们的开发板一般提供3.3V直接接上就行。GND接地和开发板共地。SIG信号输出引脚输出一个与紫外线强度成正比的模拟电压。这个引脚就是我们要用ADC去读取的地方。模块的原理是紫外线越强光电二极管产生的电流就越大经过运放放大后SIG引脚输出的电压就越高。我们通过ADC读取这个电压值就能反推出紫外线强度了。注意模块资料里提到了一个“电压-紫外线强度对照表”。这个表是厂家通过实验标定好的是咱们把ADC原始数值转换成强度等级的关键依据。后面的代码逻辑就是基于这个表来实现的。2. 硬件连接与开发板引脚配置2.1 连接传感器到开发板首先把S12SD模块和MSPM0开发板用杜邦线连起来。连接非常简单S12SD模块 VCC-开发板 3.3V引脚S12SD模块 GND-开发板 GND引脚S12SD模块 SIG-开发板 PA27引脚这里最关键的是第三根线。为什么选PA27因为我们需要一个具有ADC采集功能的引脚。在MSPM0芯片的数据手册里PA27引脚被复用了ADC功能正好可以用来采集模拟信号。你当然也可以选其他带ADC功能的引脚但代码里的配置也要相应修改。2.2 使用SysConfig工具配置引脚功能TI为MSPM0系列提供了非常方便的SysConfig图形化配置工具咱们不用再手动去查寄存器、写复杂的初始化代码了。下面跟着我一步步操作在你的CCS或IAR工程里找到并双击empty.syscfg文件或者其他你工程里的.syscfg文件它会打开SysConfig配置界面。在界面中点击Tools菜单然后选择SYSCONFIG工具确保配置工具被激活。我们需要添加两个配置一个用于ADC一个用于我们连接传感器的GPIO引脚PA27。添加ADC配置在左侧的“Peripherals”或“Drivers”中找到ADC12把它添加到工程中。通常会有ADC12_0这样的实例。添加GPIO配置找到GPIO部分添加一个GPIO配置。然后找到PA27这个引脚将其功能Mode设置为Analog模拟输入模式。对于ADC采集引脚必须设置为模拟模式这样数字电路部分才会断开避免干扰。在ADC12的配置中我们需要将某个ADC通道例如ADC_CH0映射到我们刚刚设置的PA27引脚上。具体操作是在ADC配置里选择“Input”或“Channel Configuration”然后选择对应的通道比如ADCMEM_ADC_CH0并将其输入源Input Positive设置为对应PA27的内部连接信号例如ADC_AIN27。这个映射关系需要查一下你所用具体MSPM0型号的芯片数据手册。配置完成后点击保存按钮。重要提示保存时如果弹出一个对话框询问是否要更新或覆盖某些文件一定要选择Yes to All。这是为了让SysConfig工具生成的代码完全应用到你的工程中。保存后点击编译按钮。这里可能会遇到一些编译警告甚至错误先不用管它因为SysConfig生成代码后可能有些依赖关系还没完全处理好我们后续会包含正确的头文件来解决。完成以上步骤后SysConfig工具会自动生成一个ti_msp_dl_config.h文件里面就定义了我们刚刚配置的ADC和GPIO的所有参数。这个文件通常会被board.h包含所以我们只需要在代码里包含board.h头文件就能使用这些配置了。3. 编写传感器驱动代码硬件连好了引脚也配置了接下来就是写代码让ADC工作起来并实现强度转换。我们创建两个文件bsp_ultraviolet.c和bsp_ultraviolet.h。3.1 头文件定义 (bsp_ultraviolet.h)头文件主要做宏定义和函数声明。#ifndef _BSP_ULTRAVIOLET_H_ #define _BSP_ULTRAVIOLET_H_ #include board.h // 包含SysConfig生成的配置 // 采样次数用于求平均值滤波减少单次采样的随机误差 #define SAMPLES 30 // 函数声明 void ULTRAVIOLET_Init(void); // 初始化函数 unsigned int Get_ADC_Value(void); // 获取ADC采样值已平均 char Get_Ultraviolet_Intensity(uint16_t value); // 将ADC值转换为强度等级 #endif3.2 初始化与ADC采集函数 (bsp_ultraviolet.c)这是核心部分我们分段讲解。首先包含必要的头文件并定义一个全局标志位用于在中断中通知主程序ADC转换完成。#include bsp_ultraviolet.h #include stdio.h volatile bool gCheckADC; // ADC采集成功标志位初始化函数这个函数目前主要就是开启ADC的中断。ADC转换完成后通过中断通知我们这是一种高效的方式避免了CPU一直轮询等待。void ULTRAVIOLET_Init(void) { // 开启ADC中断ADC12_0_INST_INT_IRQN这个中断号在ti_msp_dl_config.h中定义 NVIC_EnableIRQ(ADC12_0_INST_INT_IRQN); }单次ADC读取函数这个函数负责启动一次ADC转换并等待转换完成然后读取结果。uint32_t ADC_GET(void) { unsigned int gAdcResult 0; int timeout 20; // 超时计数器防止卡死 // 软件触发ADC开始一次转换 DL_ADC12_startConversion(ADC12_0_INST); // 等待中断服务函数将gCheckADC标志置为true while (false gCheckADC) { delay_us(1); timeout--; // 如果等待超时打印错误信息并返回 if(timeout 0) { printf(DL_ADC12_startConversion ERROR!! LINE:%d\r\n,__LINE__); return 1; } } // 转换成功从ADC结果存储器MEM0中读取数据 // ADC12_0_ADCMEM_ADC_CH0 对应我们之前配置的通道 gAdcResult DL_ADC12_getMemResult(ADC12_0_INST, ADC12_0_ADCMEM_ADC_CH0); // 清除标志位为下一次转换做准备 gCheckADC false; return gAdcResult; }平均值滤波函数ADC采样容易受到噪声干扰一次读数可能不准。常见的做法是连续采样多次然后取平均值。unsigned int Get_ADC_Value( void ) { uint32_t Data 0; // 循环采样 SAMPLES 次这里是30次 for(int i 0; i SAMPLES; i) { Data ADC_GET(); // 累加每次的采样值 delay_ms(5); // 每次采样间隔5ms让信号稳定一下 } Data Data / SAMPLES; // 计算平均值 return Data; }ADC中断服务函数当ADC转换完成时芯片会自动跳转到这个函数。我们在这里检查中断来源如果是我们关心的“结果已加载”中断就把标志位置1。void ADC12_0_INST_IRQHandler(void) { // 查询是哪个ADC中断触发了 switch (DL_ADC12_getPendingInterrupt(ADC12_0_INST)) { // 检查是否是我们配置的存储器0MEM0完成了数据采集 case DL_ADC12_IIDX_MEM0_RESULT_LOADED: gCheckADC true; // 将标志位置1通知主循环 break; default: break; } }3.3 核心电压值到强度等级的转换这是整个项目的“大脑”。我们需要把ADC读到的原始数字量转换成有实际意义的紫外线强度等级。转换依赖厂家提供的“电压-强度对照表”。在代码里我们用一系列if条件判断来实现这个对照表。假设我们的ADC是12位的0-4095参考电压是3.3V。那么ADC数值和电压的对应关系是电压值 (ADC数值 / 4095) * 3.3V。厂家给的对照表是电压值但我们可以直接换算成ADC的数值边界在代码里进行比较这样效率更高。下面的函数Get_Ultraviolet_Intensity就实现了这个逻辑。它接收一个ADC平均值然后判断它落在哪个区间返回对应的0-11级强度。char Get_Ultraviolet_Intensity(uint16_t value) { char ret 0; // 以下阈值是根据厂家电压表结合3.3V参考电压和12位ADC换算得出 if( value 227 ) //紫外线强度0级 { ret 0; } if( value 227 value 318 ) //紫外线强度1级 { ret 1; } if( value 318 value 408 ) //紫外线强度2级 { ret 2; } if( value 408 value 503 ) //紫外线强度3级 { ret 3; } if( value 503 value 606 ) //紫外线强度4级 { ret 4; } if( value 606 value 696 ) //紫外线强度5级 { ret 5; } if( value 696 value 795 ) //紫外线强度6级 { ret 6; } if( value 795 value 881 ) //紫外线强度7级 { ret 7; } if( value 881 value 976 ) //紫外线强度8级 { ret 8; } if( value 976 value 1079 ) //紫外线强度9级 { ret 9; } if( value 1079 value 1170 ) //紫外线强度10级 { ret 10; } if( value 1170 ) //紫外线强度11级 { ret 11; } return ret; }提示这里的阈值227 318 408...是示例代码中给出的。在实际项目中你需要根据你使用的具体开发板的ADC参考电压、以及传感器模块资料里精确的电压-强度对照表重新计算这些阈值。计算方法是ADC阈值 (对照表电压 / 参考电压) * ADC满量程值。4. 主程序集成与验证最后我们在主函数里把上面的模块调用起来并打印出结果进行验证。#include board.h #include stdio.h #include bsp_ultraviolet.h int main(void) { uint16_t value 0; // 开发板初始化时钟、串口等 board_init(); // 初始化紫外线传感器模块主要是开启ADC中断 ULTRAVIOLET_Init(); printf(UV Sensor demo start\r\n); while(1) { // 1. 获取ADC采样平均值 value Get_ADC_Value(); // 2. 将ADC值转换为强度等级 // 3. 通过串口打印出强度等级 printf(UV Intensity Grade %d\r\n, Get_Ultraviolet_Intensity( value ) ); // 延时1秒再进行下一次测量 delay_ms(1000); } }将代码编译下载到开发板打开串口调试助手波特率根据你的board_init配置通常是115200。把传感器放在室内你应该能看到串口持续打印出UV Intensity Grade 0因为室内的紫外线强度通常很低。如果把传感器拿到户外阳光下打印的等级数就会上升。调试小技巧如果一直打印0级可以先用万用表测量一下SIG引脚对地的电压看看在光照下是否有变化。确保硬件连接正确。可以同时打印出原始的value(ADC平均值)看看数值是否随着光照变化这有助于你验证ADC采集部分是否正常以及校准你的强度等级阈值。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2411011.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!