杰理AC6926X-多通道ADC设置

定义在adc_api.h
系统定义adc扫描的处理周期,2ms一次,每次扫描一个通道。

#if !ADKEY_SD_MULT_EN
LOOP_DETECT_REGISTER(adc_scan_detect) = {
    .time = 1,
    .fun  = adc_scan,
};
#endif

每次扫描一个通道,总共5个通道,系统默认第一个通道为AD按键通道,在下面数组变量中体现

void adc_scan()
{
    static u8 channel = 0;
    u8 next_channel;

    next_channel = channel + 1;

    if (next_channel == R_MAX_AD_CHANNEL) {
        next_channel = 0;
    }
/*    这些是无关紧要的代码块 
#if ADKEY_SD_MULT_EN    
    if (next_channel == R_AD_CH_KEY) {
        if (adkey_sd_mult_sd_suspend() == true) {
            key_ad_info.key_init();
            adc_value[channel++] = adc_res_api(ad_table[next_channel]);
            if (channel == R_MAX_AD_CHANNEL) {
                channel = 0;
            }
            next_channel = channel + 1;
            adc_value[channel++] = adc_res_api(ad_table[next_channel]);
            adkey_sd_mult_set_sd_io();
            adkey_sd_mult_sd_resume();
            return;
        } else {
            adc_value[channel] = adc_res_api(ad_table[next_channel + 1]); //跳过ADKEY的通道
            channel += 2;
            if (channel >= R_MAX_AD_CHANNEL) {
                channel -= R_MAX_AD_CHANNEL;
            }
            return;
        }
    }
#endif 
*/ 
    adc_value[channel++] = adc_res_api(ad_table[next_channel]); //核心处理部分
//    otp_printf("c=%d   v=%d   ",channel-1,adc_value[channel-1]);

    if (channel == R_MAX_AD_CHANNEL) {
        channel = 0;
    }
}

ad转换函数,输入采集通道,返回通道采集值

u16 adc_res_api(u32 channel)
{
    u16 adc_value = 0;
    u16 adc_con_tmp = 0;

    while (!(BIT(7) & JL_ADC->CON));	  //wait pending
    adc_value = JL_ADC->RES;

    adc_con_tmp = JL_ADC->CON & (~0x0F00);
    JL_ADC->CON = adc_con_tmp | (channel & 0xFFFF);//channel; //AD channel select
    if ((channel & 0xFFFF) == AD_CH_RTC) {
        adc_mux_ch_set((channel & 0xFFFF0000) >> 16);
    } else if ((channel & 0xFFFF) == AD_CH_PMU) {
        adc_ldo_ch_sel((channel & 0xFFFF0000) >> 16);
    } else if ((channel & 0xFFFF) == AD_CH_BT) {
        WLA_CON0 |= ((0x1 & 0X1) << 0) | \
                    ((0x0 & 0X1) << 1) | \
                    ((0x1 & 0X3) << 2) | \
                    ((0x08 & 0X1f) << 6) | \
                    ((0x2 & 0x3) << 11) | \
                    ((0x8 & 0xf) << 13);
    }
    JL_ADC->CON |= BIT(6);             //AD start

    return adc_value;
}

AC6926X一共定义了16个通道 位于JL_ADC->CON的bit[8-11]

//AD channel define JL_ADC->CON的bit[8-11]
#define AD_CH_PA3    (0x0<<8)
#define AD_CH_PA4    (0x1<<8)
#define AD_CH_PA5    (0x2<<8)
#define AD_CH_PA6    (0x3<<8)
#define AD_CH_PC4    (0x4<<8)
#define AD_CH_PA10   (0x5<<8)
#define AD_CH_PB0    (0x6<<8)
#define AD_CH_PB1    (0x7<<8)
#define AD_CH_PB4    (0x8<<8)
#define AD_CH_PB5    (0x9<<8)
#define AD_CH_PC3    (0xA<<8)
#define AD_CH_DM     (0xB<<8)
#define AD_CH_RTC    (0xC<<8)
#define AD_CH_PMU    (0xD<<8)
#define AD_CH_BT     (0xE<<8)
#define AD_CH_AUDIO  (0xF<<8)

#define ADC_RTC_MUX_PR1       (0x1<<16)
#define ADC_RTC_MUX_PR2       (0x2<<16)
#define ADC_RTC_MUX_RTC_HOSC  (0x3<<16)
#define ADC_RTC_MUX_RTC_LOSC  (0x4<<16)
#define ADC_RTC_MUX_RTCVDD    (0x5<<16)

#define ADC_PMU_MUX_VBG       (0x0<<16)
#define ADC_PMU_MUX_VDD       (0x1<<16)
#define ADC_PMU_MUX_VDDDA     (0x2<<16)
#define ADC_PMU_MUX_VDC13     (0x3<<16)
#define ADC_PMU_MUX_VBAT      (0x4<<16)
#define ADC_PMU_MUX_LDOIN     (0x5<<16)
#define ADC_PMU_MUX_RTCLDO33  (0x6<<16)
#define ADC_PMU_MUX_CHARGE    (0x7<<16)

#define AD_CH_PR1       (AD_CH_RTC | ADC_RTC_MUX_PR1)
#define AD_CH_PR2       (AD_CH_RTC | ADC_RTC_MUX_PR2)
#define AD_CH_RTC_HOSC  (AD_CH_RTC | ADC_RTC_MUX_RTC_HOSC)
#define AD_CH_RTC_LOSC  (AD_CH_RTC | ADC_RTC_MUX_RTC_LOSC)
#define AD_CH_RTCVDD    (AD_CH_RTC | ADC_RTC_MUX_RTCVDD)

枚举变量用于定义每个变量的名字 例如adc_value[R_AD_CH_KEY] = adc_value[0]
而对应的ad_table[]则保存对应通道的JL_ADC->CON的bit[8-11]值。
自行添加修改通道是时,需要对应2个数组的位置,以免读错通道值,所有通道扫描一次时间2ms*最大通道数,如果通道太多,ad按键就需要微调,不然可能会有不灵敏的感觉

/*AD通道定义*/
enum {
#if KEY_AD_RTCVDD_EN
    R_AD_CH_KEY = 0,
    R_AD_CH_RTCVDD,
#elif KEY_AD_VDDIO_EN
    R_AD_CH_KEY = 0,
#endif
#if POWER_EXTERN_DETECT_EN
    R_AD_CH_EXTERN_POWER,
#endif
 //   R_AD_CH_VBAT,
 //   R_AD_CH_CHARGE,
    R_AD_CH_temporary, //PB1
    R_AD_CH_battery,   //PR1
    // R_AD_CH_LDOIN,
    R_AD_CH_LDOREF,
    R_AD_CH_BT,
    R_MAX_AD_CHANNEL,
};

adc_api.c下定义了两个数组只有5个元素,adc_value[]

u16 adc_value[R_MAX_AD_CHANNEL];
const u32 ad_table[] = {
#if KEY_AD_RTCVDD_EN
    AD_KEY_CH,
    AD_CH_RTCVDD,
#elif KEY_AD_VDDIO_EN
    AD_KEY_CH,
#endif
#if POWER_EXTERN_DETECT_EN
    AD_POWER_EXTERN_CH,
#endif
    //AD_CH_VBAT,
    // AD_CH_CHARGE,
    AD_CH_PB1, 
    AD_CH_PR1, 
    /* AD_CH_LDOIN, */
    AD_CH_LDOREF,
    AD_CH_BT, // 1110 ,BT to ADC
};

操作很简单,只需要修改制定的通道及名称就可以,在需要使用到的地方调用数组的值即可,例如SDK的ad按键部分,函数体中直接取出对应的数组值。
key_value =adc_value[R_AD_CH_KEY];

u8 get_adkey_value(void)
{
    u8 key_number;
    u32 key_value;
    static u8 aux_in_cnt = 0;
    static u8 aux_out_cnt = 0;

    key_value = adc_value[R_AD_CH_KEY];

后面一些特殊的ADC通道不明确有什么特殊的配置需要,普通IO口的采样直接配置数组就可以,配置好后,在上面的的枚举和数组中添加对应的ad通道号就行了

	// PB1 配置AD输入  ADC7
    JL_PORTB->PU  |= BIT(1);    //使用内部上拉要打开上拉
    JL_PORTB->PD  &= ~BIT(1);
    JL_PORTB->DIR |=  BIT(1); //GPIO设置输入模式
    JL_PORTB->DIE &= ~BIT(1); //GPIO输入关闭,使用ADC模式
    //PR1 adc12 配置AD输入
    PORTR_PU (PORTR1, 1);
    PORTR_PD (PORTR1, 0);
    PORTR_DIR(PORTR1, 1);
    PORTR_DIE(PORTR1, 0);
    PORTR1_ADCEN_CTL(1);

你可能感兴趣的:(杰理AC692x学习)