GD32F105RC 的PA0 (ADC_IN0) 采样电压值偏大

项目场景:

  使用GD32F105RC 的 ADC_IN0 外接10k热敏电阻测量温度,发现测量误差较大。下图为硬件电路。


 

问题描述

初始化ADC_IN0

/**=====================================================
  * @brief   初始化ADC_1(PA1), 利用 PTC 获取温度 (通过 CD4051 循环切换6个测温通道)
  * @param   None
  * @retval  None
  =====================================================**/
void ADC1_Init(void)
{ 	
	ADC_InitTypeDef ADC_InitStructure; 
	GPIO_InitTypeDef GPIO_InitStructure;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1, ENABLE );	  //使能ADC1通道时钟
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);                   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M

	/*--------- Temp_ADC--PA1 ------------------------------------------------*/        
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;		//模拟输入引脚   
	GPIO_Init(GPIOA, &GPIO_InitStructure);	
	ADC_DeInit(ADC1);   //复位ADC1 

	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;	//ADC工作模式:ADC1和ADC2工作在独立模式
	ADC_InitStructure.ADC_ScanConvMode = DISABLE;     	//模数转换工作在单通道模式
	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;	//模数转换工作在单次转换模式
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;	//软件触发启动转换
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;	  //ADC数据右对齐
	ADC_InitStructure.ADC_NbrOfChannel = 1;	//顺序进行规则转换的ADC通道的数目
	ADC_Init(ADC1, &ADC_InitStructure);	//根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器   

  
	ADC_Cmd(ADC1, ENABLE);	//使能指定的ADC1
	
	ADC_ResetCalibration(ADC1);		//使能复位校准  
	 
	while(ADC_GetResetCalibrationStatus(ADC1));//等待复位校准结束
	
	ADC_StartCalibration(ADC1);	  //开启AD校准
 
	while(ADC_GetCalibrationStatus(ADC1));	  //等待校准结束
 
}	

 获取PA0端口电压,计算出温度。

/**=====================================================
  * @brief   获得ADC通道1的值(PA1), 
  * @param   ch :ADC通道号,通道值 0~3
  * @note   
  * @retval  返回测量到的电压值 ADC_GetConversionValue(ADC1)
  =====================================================**/
u16 Get_Adc(u8 ch)   
{
  //设置指定ADC的规则组通道,一个序列,采样时间
	ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 );	//ADC1,ADC通道,采样时间为239.5周期	 	    
  
	ADC_SoftwareStartConvCmd(ADC1, ENABLE);		//使能指定的ADC1的软件转换启动功能	
	 
	while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束

	return ADC_GetConversionValue(ADC1);	//返回最近一次ADC1规则组的转换结果
}

/**=====================================================
  * @brief   多次测量ADC通道值,并去平均值 
  * @param   ch    :ADC通道号
  * @param   times :测量的次数
  * @note   
  * @retval  返回多次测量平均电压
  =====================================================**/
u16 Get_Adc_Average(u8 ch,u8 times)
{
	u32 temp_val=0;
	u8 t;
	for(t=0;t<times;t++)
	{
		temp_val+=Get_Adc(ch);
		delay_ms(5);
	}
	return temp_val/times;
} 	 

/**=====================================================
  * @brief   根据测量电压值,结合 PTC 换算公式,得出温度
  * @param   
  * @note   
  * @retval  返回多次测量平均电压
  =====================================================**/
u16 get_temp(void)
{ 
	float vol[1];        /*用于存放ADC采集的电压值*/
	float Temp,Rt;
	float Rp = 10000.0;  /*热敏电阻在T2常温下的标称阻值,10k*/
	float Ka = 273.15;   /*0摄氏度*/	
	float T2 = Ka+25;    /*25摄氏度*/
	float Bx = 3950.0;   /*热敏电阻参数*/

	vol[0]=(Get_Adc_Average(ADC_Channel_0,10)&0xFFF)*3.3/4096;
	       
	                         /*获取PT100两端电压,并赋值*/
  Rt = vol[0]*10000.0 / (3.3 - vol[0]);        /*计算电阻值,这里10000.0是电路上分压电阻值*/
	Temp = 1.0/(log(Rt/Rp)/Bx + 1/T2)-Ka+0.5+30; /*计算温度,修正误差,并转化为摄氏温度*/
	return (u16)(Temp*10);
}
 

原因分析:

之前将GD32F105RCT6更换为STM32F105RCT6 后也正常了,测量值也是正常的。分析可能GD32F105RCT6内部电路不同。重新换回GD32F105RCT6后,发现是PA3输入电压超过3.3V达到了3.8V。


解决方案:

PA3的3.8V电压去掉后,PA0(ADC_IN0)采样电压值就正常。查看手册发现PA0输入电压不能大于3.3V,不兼容5V

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,这是示例代码,供参考: ``` #include "gd32f10x.h" #include <stdio.h> void ADC_GPIO_Configuration(void) { rcu_periph_clock_enable(RCU_GPIOA); gpio_init(GPIOA, GPIO_MODE_AIN, GPIO_OSPEED_50MHZ, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_3| GPIO_PIN_4|GPIO_PIN_5|GPIO_PIN_6|GPIO_PIN_7|GPIO_PIN_8|GPIO_PIN_9|GPIO_PIN_10|GPIO_PIN_11| GPIO_PIN_12|GPIO_PIN_13|GPIO_PIN_14|GPIO_PIN_15); } void ADC_Configuration(void) { rcu_periph_clock_enable(RCU_ADC0); adc_clock_config(ADC0,ADC_CLOCK_MODE_SYNCLK_DIV2); adc_mode_config(ADC_MODE_FREE); adc_special_function_config(ADC0, ADC_SCAN_MODE, ENABLE); adc_special_function_config(ADC0, ADC_CONTINUOUS_MODE, DISABLE); adc_external_trigger_source_config(ADC0, ADC_EXTTRIG_REGULAR_NONE); adc_data_alignment_config(ADC0, ADC_DATAALIGN_RIGHT); adc_channel_length_config(ADC0, ADC_INSERTED_CHANNEL, 16); adc_regular_channel_config(ADC0, 0, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 1, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 2, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 3, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 4, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 5, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 6, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 7, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 8, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 9, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 10, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 11, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 12, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 13, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 14, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(ADC0, 15, ADC_SAMPLETIME_55POINT5); adc_enable(ADC0); delay_1ms(1); } uint16_t ADC_GetValue(uint8_t rank) { uint16_t value; adc_inserted_channel_enable(ADC0, rank); delay_1ms(1); adc_software_trigger_enable(ADC0, ADC_INSERTED_CHANNEL); while(!adc_flag_get(ADC0,ADC_FLAG_EOC)); adc_flag_clear(ADC0, ADC_FLAG_EOC); value = ADC_RDATA0(ADC0); adc_inserted_channel_disable(ADC0, rank); return value; } int main(void) { uint8_t i; uint16_t adc_value; ADC_GPIO_Configuration(); ADC_Configuration(); while(1) { printf("\r\n-----------------"); for(i=0;i<16;i++) { adc_value = ADC_GetValue(i); printf("\r\nADC%d Value:%d",i,adc_value); } delay_1ms(1000); } } ``` 此代码使用了 GD32F103 的 ADC 功能,并且获得了 16 个通道的电压。你需要根据具体的需求进行适当的修改。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值