GD32VW553开发板光敏电阻传感器模块移植实战:ADC与GPIO双模式光照检测
GD32VW553开发板光敏电阻传感器模块移植实战ADC与GPIO双模式光照检测最近在做一个智能环境监测的小项目需要检测环境光照强度。手头正好有立创的GD32VW553开发板和一块常见的光敏电阻模块型号5516今天就来分享一下如何把这两者结合起来实现光照检测功能。这个教程适合正在学习GD32VW553或者需要集成光敏传感器的朋友。我会从模块原理讲起一步步带你完成硬件连接、驱动编写和代码调试让你不仅能点亮LED还能真正理解传感器的工作原理和编程方法。1. 光敏电阻模块先搞懂它怎么工作光敏电阻听起来挺专业其实原理很简单。它就是用硫化镉或硒化镉这类半导体材料做成的特殊电阻特点是光照越强电阻值越小。你可以把它想象成一个“见光怂”的电阻——有光的时候内部会产生很多能导电的粒子载流子电阻就变小了没光的时候这些粒子很少电阻就变得很大。我手头这个5516型号在光亮环境下电阻大约8-20KΩ在黑暗环境下能达到1MΩ左右。咱们用的这个模块设计得很贴心给了我们两种读取方式接口信号类型工作原理AO模拟输出模拟电压直接输出光敏电阻分压后的电压值光照越强电压越高DO数字输出数字电平通过LM393比较器电压超过阈值就输出高/低电平AO接口直接读取模拟电压需要用到MCU的ADC模数转换器功能。这样我们可以得到连续的光照强度值精度高。DO接口输出的是0或1的数字信号相当于一个“开关”。当光照低于某个阈值时输出一种状态高于阈值时输出另一种状态。这个阈值可以通过模块上的电位器调节。注意模块工作电压是3.3-5VGD32VW553开发板上有3.3V输出直接接上就行不用担心电压不匹配。2. 硬件连接把模块接到开发板上连接很简单就四根线。我选择用开发板的PB0引脚接模块的AO模拟输出PB2引脚接DO数字输出。为什么选这两个引脚因为PB0有ADC功能而PB2是普通GPIO正好满足我们的需求。具体接线如下模块引脚开发板引脚作用VCC3V3供电3.3VGNDGND接地DOPB2数字输出接GPIO输入AOPB0模拟输出接ADC输入接好线后模块上的指示灯应该会亮起。如果没亮检查一下接线有没有接反。3. 代码编写创建光照传感器驱动接下来是重头戏——写代码。我们需要创建两个文件bsp_illume.c和bsp_illume.h把它们放在你的工程里记得把包含路径添加到编译器设置中。3.1 头文件定义bsp_illume.h头文件主要是定义引脚和声明函数咱们先看看关键的宏定义#ifndef BSP_CODE_BSP_ILLUME_H_ #define BSP_CODE_BSP_ILLUME_H_ #include gd32vw55x.h #include systick.h /* 引脚定义 */ /* PB0 用作ADC输入对应ADC通道8 */ #define BSP_ADC_GPIO_RCU RCU_GPIOB #define BSP_ADC_GPIO_PORT GPIOB #define BSP_ADC_GPIO_PIN GPIO_PIN_0 #define BSP_ADC_RCU RCU_ADC #define BSP_ADC ADC #define BSP_ADC_CHANNEL ADC_CHANNEL_8 /* PB2 用作数字输入 */ #define BSP_DO_GPIO_RCU RCU_GPIOB #define BSP_DO_GPIO_PORT GPIOB #define BSP_DO_GPIO_PIN GPIO_PIN_2 /* 读取DO引脚电平的宏 */ #define BSP_DO_READ() gpio_input_bit_get(BSP_DO_GPIO_PORT, BSP_DO_GPIO_PIN) /* 函数声明 */ void illume_Init(void); // 初始化函数 int Get_illume_Adc_Value(uint8_t Count); // 获取ADC原始值 int Get_illume_Percentage_value(void); // 获取光照百分比 uint8_t Get_illume_DO_In(void); // 读取DO引脚状态 #endif /* BSP_CODE_BSP_ILLUME_H_ */这里有几个关键点BSP_ADC_CHANNEL_8PB0引脚在GD32VW553上对应的是ADC通道8这个一定要写对BSP_DO_READ()用宏定义封装了GPIO读取函数后面用起来方便函数分工明确初始化、读原始值、读百分比、读数字状态3.2 初始化函数bsp_illume.c初始化要做两件事配置ADC和配置GPIO。ADC配置稍微复杂点我一步步解释void illume_Init(void) { /* 1. 先打开时钟开关 */ rcu_periph_clock_enable(BSP_ADC_GPIO_RCU); // GPIOB时钟 rcu_periph_clock_enable(BSP_DO_GPIO_RCU); // GPIOB时钟和上面是同一个但为了清晰分开写 rcu_periph_clock_enable(BSP_ADC_RCU); // ADC时钟 /* 2. 配置DO引脚为输入模式 */ gpio_mode_set(BSP_DO_GPIO_PORT, GPIO_MODE_INPUT, GPIO_PUPD_NONE, BSP_DO_GPIO_PIN); /* 3. 配置ADC时钟为4分频 */ adc_clock_config(ADC_ADCCK_PCLK2_DIV4); /* 4. 配置ADC引脚为模拟输入模式 */ gpio_mode_set(BSP_ADC_GPIO_PORT, GPIO_MODE_ANALOG, GPIO_PUPD_NONE, BSP_ADC_GPIO_PIN); /* 5. 配置ADC工作模式 */ adc_special_function_config(ADC_SCAN_MODE, ENABLE); // 使能扫描模式 adc_data_alignment_config(ADC_DATAALIGN_RIGHT); // 数据右对齐 adc_resolution_config(ADC_RESOLUTION_12B); // 12位分辨率 /* 6. 配置ADC通道 */ adc_channel_length_config(ADC_ROUTINE_CHANNEL, 1); // 规则组用1个通道 /* 7. 配置触发方式 */ adc_external_trigger_config(ADC_ROUTINE_CHANNEL, EXTERNAL_TRIGGER_DISABLE); // 禁用外部触发 adc_software_trigger_enable(ADC_ROUTINE_CHANNEL); // 使能软件触发 /* 8. 最后使能ADC */ adc_enable(); }这里有个坑我踩过ADC的配置顺序很重要。一定要先配置时钟和引脚模式再配置ADC参数最后才使能ADC。如果顺序乱了ADC可能工作不正常。提示GPIO_MODE_ANALOG是模拟输入模式当引脚用作ADC输入时必须设置成这个模式不能设置成普通的输入模式。3.3 ADC读取函数ADC配置好后怎么读取数据呢我写了一个ADC_GET()函数static int ADC_GET(void) { uint16_t gAdcResult 0; uint16_t timeOut 1000; // 超时计数器 /* 设置要采集的通道 */ adc_routine_channel_config(0, BSP_ADC_CHANNEL, ADC_SAMPLETIME_14POINT5); /* 启动转换 */ adc_software_trigger_enable(ADC_ROUTINE_CHANNEL); /* 等待转换完成 */ while(!adc_flag_get(ADC_FLAG_EOC) timeOut--) { delay_us(1); } /* 清除完成标志 */ adc_flag_clear(ADC_FLAG_EOC); if(!timeOut) { printf(ADC_GET Failed!!!\r\n); return 0; } /* 读取转换结果 */ gAdcResult adc_routine_data_read(); return gAdcResult; }这里有几个细节ADC_SAMPLETIME_14POINT5是采样时间单位是时钟周期。14.5个周期对于光敏电阻这种变化不快的信号足够了。一定要加超时判断如果ADC卡住了程序不会一直死等。读取数据后要清除标志位不然下次转换可能出问题。3.4 获取光照百分比直接看ADC原始值0-4095不够直观我更喜欢用百分比表示光照强度。0%表示最暗100%表示最亮int Get_illume_Percentage_value(void) { // ADC是12位的最大值是2^12 - 1 4095 int adc_max 4095; int adc_new 0; int Percentage_value 0; // 读取5次取平均减少波动 adc_new Get_illume_Adc_Value(5); // 计算百分比 // 注意光照越强ADC值越小所以用1减去比值 Percentage_value ( 1 - ( (float)adc_new / adc_max ) ) * 100; return Percentage_value; }这里有个关键点光照越强光敏电阻阻值越小分压后电压越低ADC值就越小。所以ADC值越小代表光照越强。我在计算百分比时用1 - (adc_new / adc_max)来反转这个关系。Get_illume_Adc_Value(5)函数会读取5次ADC值然后取平均这样数据更稳定。实际使用中你可以根据需求调整采样次数。3.5 读取数字输出数字输出就更简单了直接读取GPIO电平uint8_t Get_illume_DO_In(void) { if( BSP_DO_READ() 1 ) { return 1; // 检测到过暗 } return 0; // 光照正常 }模块上的LM393比较器会对比光敏电阻的电压和一个可调阈值。当光照较暗电压低于阈值时DO输出高电平1光照较亮时输出低电平0。你可以通过调节模块上的蓝色电位器来改变这个阈值。4. 主程序测试驱动写好了在主程序里测试一下。在main.c中添加#include bsp_illume.h int main(void) { /* 系统初始化时钟、串口、LED等 */ systick_config(); gd_eval_led_init(LED1); gd_eval_com_init(EVAL_COM0); /* 初始化光照传感器 */ illume_Init(); printf(Illumination sensor initialized.\r\n); while(1) { /* 读取并打印所有数据 */ printf(ADC Value %d\r\n, Get_illume_Adc_Value(5)); printf(Percentage value %d%%\r\n, Get_illume_Percentage_value()); printf(DO %d\r\n, Get_illume_DO_In()); delay_ms(500); // 每500ms读取一次 } }编译下载后打开串口调试助手波特率根据你的配置设置应该能看到类似这样的输出ADC Value 1245 Percentage value 70% DO 0用手遮住光敏电阻或者用手机手电筒照它看看数值变化是否正常。5. 实际使用中的几个要点根据我的项目经验有几点需要注意1. ADC采样稳定性光敏电阻的响应不是瞬间的有一定惯性。如果你发现ADC值跳动比较大可以增加采样次数比如从5次增加到10次在软件里做滑动平均滤波适当增加ADC采样时间2. DO阈值调节模块上的蓝色电位器可以调节比较器的阈值。顺时针旋转阈值电压升高需要更强的光照才能让DO输出低电平逆时针旋转则相反。根据你的应用场景调节到合适的位置。3. 光照百分比校准我的百分比计算是基于ADC最大值的理论计算。在实际应用中你可能需要校准在最暗环境下读取ADC值作为0%参考在最亮环境下读取ADC值作为100%参考然后用这两个值做线性映射4. 功耗考虑这个模块工作电流约1mA对于电池供电项目来说不算大。但如果需要长时间待机可以考虑定时采样而不是连续采样。6. 扩展应用思路这个基础驱动做好后你可以做很多有趣的应用自动调光台灯根据环境光照自动调节LED亮度光照暗时调亮光照强时调暗。智能窗帘控制光照太强时自动关闭窗帘光照适中时半开早晨光照足够时自动拉开。植物补光系统检测植物所在位置的光照不足时自动开启补光灯。光照数据记录结合RTC实时时钟记录一天中不同时间的光照变化分析光照规律。代码我已经在实际项目中用过稳定性没问题。如果你在移植过程中遇到问题重点检查以下几点引脚定义是否正确PB0对应ADC通道8时钟是否使能ADC配置顺序是否正确串口打印是否正常好了关于GD32VW553驱动光敏电阻模块的内容就分享到这里。实际动手做一遍你会发现并没有想象中那么难。嵌入式开发就是这样多看数据手册多动手调试慢慢就熟练了。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2410820.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!