STM8L151系列 单通道ADC数模转换

IAR软件,使用官方halt库

 

void insertion_sort(u16 a[], u16 length)//插入排序
{
    int len = length;
    int temp;
    int i;//已排好序的序列的下一个元素(待插元素)的下标 
    int j;//有序序列的末尾下标

    for(i = 1;i<len;i++)
    {
        temp = a[i];//中间变量 
        j=i-1;
        while(j >= 0 && a[j]>temp){//遍历有序序列,与要插入的元素比较 
            a[j+1] = a[j];//将元素后移,满足条件后移一个位置
            j--; 
        }
        a[j+1] = temp; //插入  
    }


    printf("after_insert: \r\n");
    for(i = 0 ; i< len; i++)
    {
        DBG_PRINT("%d ",a[i]);
    }
    printf("\r\n");
}

/*
https://blog.csdn.net/Leo_Luo1/article/details/77370818
    另外STM8L内部还提供了1.2V的参考电压,但是在使用这个参考电压的时候有一些不稳定的现象。ADC_ChannelCmd()
    比如在采集这个参考电压时,如果在读取函数前面加延时,延时的时间不一样得到的采样结果是不一样的,
    另外,这个参考我根据外面提供的VCC作为参考电压来计算的话,得到的值也不是1.2V,这个地方不知道是我的配置出了问题还是这个参考电压本身有问题。
datasheet中写道以下一句话用于解释为什么你的第一次采样不对
The ADC uses the internal reference voltage, so the user must make sure this reference voltage is woken up before enabling the ADC after an exit from Halt or Active-halt mode. 
When enabled, the ADC needs a stabilization time (tWKUP) before starting the first conversion (refer to the device datasheet) Example: after an RTC wakeup (alarm) from Halt mode, 
the ADC is first enabled. Then, the firmware waits for a tWKUP time (3 μs) and the conversion can be started by setting the START bit in the ADC_CR1 register. 
也就是在ref电压稳定后,大概需要等待3us
*/

void init_adc(ADC_Channel_TypeDef ADC_Channel_x)
{
    ADC_Init(ADC1, ADC_ConversionMode_Single, ADC_Resolution_12Bit, ADC_Prescaler_2);//单次转换,12位分辨率(精度),ADC_clock: 4 /2 =2MHz

//    <Slow Channels group(Channel 0 to 23)
//    <Fast Channels group Channel 24 to 27)
    ADC_SamplingTimeConfig(ADC1, ADC_Group_SlowChannels, ADC_SamplingTime_4Cycles); //ADC_clock: 2MHz; 配置ADC为快速通道,采样时间周期为9;  {4; 9; 16; 24; 48; 96; 192; 384;}

    ADC_Cmd(ADC1, ENABLE); //使能ADC1
    ADC_ChannelCmd(ADC1, ADC_Channel_x, ENABLE);//使能 通道x

//    ADC_ChannelCmd(ADC1, ADC_Channel_Vrefint, ENABLE);  //内部参考电压


    ADC_ITConfig(ADC1, ADC_IT_EOC, DISABLE); //Disable End of conversion ADC1 Interrupt
    ADC_DMACmd(ADC1,DISABLE); // 禁止DMA

    ADC_SoftwareStartConv (ADC1);//开始ADC软件转换
}


unsigned short adc_handling(ADC_Channel_TypeDef ADC_Channel_x)
{
    u16 get_value = 0;

    while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC)) ;//等待转换结束
    ADC_ClearFlag (ADC1, ADC_FLAG_EOC);//清除对应标志

    get_value = ADC_GetConversionValue (ADC1);//获取转换值

    return get_value;
}


unsigned short adc_common(GPIO_TypeDef* GPIOx, uint8_t GPIO_Pin, ADC_Channel_TypeDef adc_channel)
{
#define ADC_COLLECTION_COUNT    10
    u8 i = 0;
    long sum = 0;
    u16 value=0, tmp_value[ADC_COLLECTION_COUNT] = {0};

    GPIO_Init(GPIOx, GPIO_Pin, GPIO_Mode_In_FL_No_IT);
    CLK_PeripheralClockConfig(CLK_Peripheral_ADC1, ENABLE);//开启ADC时钟

    for(i=0; i<ADC_COLLECTION_COUNT; i++)
    {
        init_adc(adc_channel);
        tmp_value[i] = adc_handling(adc_channel);
    }

    // 应该在操作ADC的寄存器关停它之后,才停止它的时钟。
    ADC_ChannelCmd(ADC1, adc_channel, DISABLE);//禁用 通道x
    ADC_Cmd(ADC1, DISABLE);
    CLK_PeripheralClockConfig(CLK_Peripheral_ADC1, DISABLE);// 禁用ADC时钟

//    GPIO_Init(GPIOx, GPIO_Pin, GPIO_Mode_Out_PP_Low_Slow); //不要设置输出模式;  悬空输入就是 低功耗
    insertion_sort(tmp_value, ADC_COLLECTION_COUNT); // 排序: 去掉 最大值,最小值; 在求平均值
    for(i=1; i<ADC_COLLECTION_COUNT - 1; i++)
        sum += tmp_value[i];

    value = sum / (ADC_COLLECTION_COUNT - 2);
    return value;
}

unsigned short get_adc(void)
{
    return adc_common(GPIOD, GPIO_Pin_2, ADC_Channel_20); //  xx电阻的ADC; PD^2; ADC1_IN20(通道20) 
}

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值