1、读 datasheet

在《DS_TLSR8267-E21_Datasheet for Telink BLE SoC TLSR8267.pdf》第11章详细介绍了ADC相关属性及参数。


2、看封装好驱动

在sdk/proj/mcu_spec/adc_8267.c中实现了8267 adc的封装,对外函数比较简单:

  • (普通12个channels初始化) void adc_Init(enum ADCCLOCK adc_clk,enum ADCINPUTCH chn,enum ADCINPUTMODE mode,enum ADCRFV ref_vol, enum ADCRESOLUTION resolution,enum ADCST sample_cycle)
  • (电池检测初始化)void adc_BatteryCheckInit(enum ADCCLOCK adc_clk,unsigned char div_en,enum ONETHIRD_INPUTCHN oneThirdChn,enum ADCINPUTCH notOneThirdChn, enum ADCINPUTMODE mode,enum ADCRFV ref_vol,enum ADCRESOLUTION resolution,enum ADCST sample_cycle)
  • (温度检测初始化)void adc_TemSensorInit(enum ADCCLOCK adc_clk,enum ADCINPUTCH chn,enum ADCINPUTMODE mode,enum ADCRFV ref_vol, enum ADCRESOLUTION resolution,enum ADCST sample_cycle)
  • (读取ADC数据)unsigned short adc_SampleValueGet(void)
  • (读取电池电量数据)unsigned short adc_BatteryValueGet(void)

用起来比较简单,初始化,然后读取!对于初始化中的枚举参数,可以看看定义处,就明白了。下面举个简单的B6通道读取数据的例子:

adc_Init(ADC_CLK_4M, B6, SINGLEEND, RV_AVDD, RES14, S_6);

while(1<2){
    u16 ret = adc_SampleValueGet();
}


3、计算实际数值

在/sdk/vendor/826x_driver_test/app_adc.c写了一个电池电量读取的DEMO,里面初始化和数据读取和我们第二节介绍的大同小异,其中有个读取的ret值转换为实际电压值的小程序片段:

switch(ref_vol){
    case RV_1P428:
    #if (BATT_CHECK_ENABLE)
        app_adc_test_Vol = 3*(1428*(average_data-128)/(16383-256)); //2^14 - 1 = 16383;
    #else
        app_adc_test_Vol = 1428*(average_data-128)/(16383-256); //2^14 - 1 = 16383;
    #endif
    break;
case RV_AVDD:
    #if (BATT_CHECK_ENABLE)
        app_adc_test_Vol = 3*(3300*(average_data-128)/(16383-256)); //2^14 - 1 = 16383;
    #else
        app_adc_test_Vol = 3300*(average_data-128)/(16383-256); //2^14 - 1 = 16383;
    #endif
    break;
case RV_1P224:
    #if (BATT_CHECK_ENABLE)
        app_adc_test_Vol = 3*(1224*(average_data-128)/(16383-256)); //2^14 - 1 = 16383;
    #else
        app_adc_test_Vol = 1224*(average_data-128)/(16383-256); //2^14 - 1 = 16383;
    #endif
    break;
}

其中由于分辨率为RES14=14bits=2^14-1


4、设计读取两个channels的值

从第二节我们知道初始化后直接可以loop读取ADC值,那如果我想要同时读取两个channels的ADC该怎么办呢?

通过观察sdk/proj/mcu_spec/adc_8267.c中的adc_Init发现,有一个static inline函数可以切换channel:

/********************************************************
*
*   @brief      set ADC analog input channel
*
*   @param      adcCha - enum variable adc channel.
*               adcInCha - enum variable of adc input channel.
*
*   @return     None
*/
static inline void adc_AnaChSet(enum ADCINPUTCH adcInCha){
    unsigned char cnI;

    cnI = (unsigned char)adcInCha;
    BM_CLR(reg_adc_chn_m_sel,FLD_ADC_CHN_SEL);
    reg_adc_chn_m_sel |= MASK_VAL(FLD_ADC_CHN_SEL,cnI);
}

那么能否想要读取B6 channel时候切换到B6读取,想要读取B4的时候切换到B4?所以先写个代码试试:

void my_adc_init(void){
    adc_Init(ADC_CLK_4M, B6, SINGLEEND, RV_AVDD, RES14, S_6);
}

void adc_AnaChSet(enum ADCINPUTCH adcInCha){
    unsigned char cnI;

    cnI = (unsigned char)adcInCha;
    BM_CLR(reg_adc_chn_m_sel,FLD_ADC_CHN_SEL);
    reg_adc_chn_m_sel |= MASK_VAL(FLD_ADC_CHN_SEL,cnI);
}

void my_adc_run(void){
    u16 ret;
    u8 res;

    static u16 cnt = 0;

    if(cnt == 0 || cnt == 4000){
        adc_AnaChSet(B6);
        cnt = 0;
    }else if(cnt == 2000){
        adc_AnaChSet(B4);
    }
    cnt ++;

    ret = adc_SampleValueGet();
    res = 3300*(ret-128)/(16383-256)/100;//100mV
    my_uart_send_data(&res,1);
}

起初,我直接俄高频切换、读取、再切换、再读取,会出现意想不到的结果,推测可能是切换和读取太快,频率快于AUTO模式的数据采集了,因此用一个cnt分段切换通道读取,发现效果很好。


5、重要常用输入资源表

有了下面的枚举类型,很容易看出8267的ADC输入资源情况,方便选择正确的IO口:

//ADC analog input channel selection enum
enum ADCINPUTCH{
    NOINPUT,
    C0,//GPIO_PC0
    C1,
    C6,
    C7,
    B0,
    B1,
    B2,
    B3,
    B4,
    B5,
    B6,
    B7,
    PGAVOM,
    PGAVOP,
    TEMSENSORN,
    TEMSENSORP,
    AVSS,
    OTVDD,//1/3 voltage division detection
};



: 蓝牙芯片:国产芯片的技术博客比外国芯片少一些,经常查不到~
: 大家觉得不错,可以点推荐给更多人~


LINKS

[1]. telink官网
[2]. 搭建tlsr8266编译框架在win服务器中


@beautifulzzzz
以蓝牙技术为基础的的末梢无线网络系统架构及创新型应用探索!
领域:智能硬件、物联网、自动化、前沿软硬件
博客:https://www.cnblogs.com/zjutlitao/
微信交流群|微信:园友交流群|btfzzzz
10-12 12:53