NRF52832学习笔记(7)——ADC接口使用

一、简介

1.1 主要特点

NRF52832 中 ADC 为一个 逐次逼近(SAADC) 模拟数字转换器

  • 8/10/12 位分辨率,采用过采样可以达到 14 位分辨率。
  • 多达 8 个输入通道:
    单端输入时有 1 个通道,2 个通道组成差分输入。
    单端和差分输入时可以配置成扫描模式。
  • 满量程输入范围为 0 和 VDD
  • 可以通过软件触发采样任务启动采样,也可以使用低功耗 32.768KHz 的 RTC 定时器或更精确的 1/16MHz 定时器通过 PPI 来触发采样任务。
  • NRF52832 的 SAADC 支持 单次模式和扫描模式
    单次的采集模式只使用一个采集通道。
    扫描模式是按照顺序采样一系列通道。Sample delay between channels is tack + tconv
    which may vary between channels according to user configuration of tack.
  • 通过 EasyDMA 可以直接将采样结果保存到 RAM。
  • 无需外部定时器即可实现连续采样。
  • 可配置通道输入负载电阻。
  • 具备采样值门限检测功能。

1.2 采样模式

NRF52832 的 ADC 有 16 个通道(其中 8个正端输入通道P ,和 8个负端输入N),因此信号的采样模式可分为单端输入和差分输入。


默认状态下,ADC 的配置模式为 单端输入 (即 CH[n].CONFIG 寄存器中的 MODE = 0),此时芯片内部将 ADC 负极输入短接到地。

差分输入 (即 CH[n].CONFIG 寄存器中的 MODE = 1),则是把负极通过负向端输入,通过计算两端的差值来换算出采样结果。

使用单端模式时,是将内部地和外部待测电压的参考地假设为一样来考虑的,但是地弹噪声会导致 ADC 产生误差,如果这个误差超过我们能接受的范围,建议使用差分输入。

在参考源上,相对于 nRF51822,nRF52832 取消了外部参考源,只能使用内部参考源。

ADC 输出的采样取决与 CH[n].CONFIG 和 RESOLUTION 寄存器配置的参数,采样结果计算公式如下:

其中:

  • V(P):ADC 输入正极。
  • V(N):ADC 输入负极。
  • GAIN:CH[n].CONFIG 寄存器中设置的增益 GAIN(1/6、1/5、1/4、1/3、1/2、1、2、4)。
  • REFERENCE:参考电压(两种方式:一个是内部参考电压0.6V,另一个是VDD/4为参考电压)。
  • RESOLUTION:采样精度(8/10/12,采用过采样可以达到 14 位)。
  • m:如果 ADC 配置为单端模式,m = 0,如果 ADC 配置为差分模式,m = 1。

1.3 工作模式

SAADC 有三种工作模式:单次转换、连续转换和扫描模式。

1.3.1 单次转换

配置ADC的寄存器CH[n].PSELP、CH[n].PSELN 和 CH[n].CONFIG 使得ADC工作于单次模式。触发采样任务后,ADC 开始采样输入电压,采样时间通过 CH[n].CONFIG.TACQ 配置。EVENTS_DONE 事件表明了一次采样的完成。
在没有过采样发生的情况下 EVENTS_RESULTDONE 事件等同于 EVENTS_DONE 事件,注意在实际采样数据通过EasyDMA 被保存到 RAM 之前,这两个事件都会产生。

1.3.2 连续转换

可以通过下面两种方式实现连续模式:

  • 使用 ADC 内部定时器实现定时采样,ADC 有一个 SAMPLERATE 寄存器,该寄存器可以配置为 Timer,配置 SAMPLERATE 的比较值 SAMPLERATE.CC 即可实现定时采样。这种方式下,触发一次采样任务即可对所有使能的通道进行采样。
  • 使用 nRF52832 的通用定时器定时通过 PPI 触发采样,实现连续采样。这种方式不能算是 ADC 本身的连续采样功能,因为它是借助了其他外设实现的连续采样。
     采样速率由 SAMPLERATE.CC 控制,即定时时间长,采样速率慢,定时时间短,采样速率快,需要注意的是,使用连续采样模式的时候,采样速率应符合下面的公式:

使用 SAMPLERATE 的 Timer 实现的连续采样不能和扫描模式结合使用,连续采样模式下只能使用一个通道。
EVENTS_DONE 事件表明了一次采样的完成,连续采样模式下,在没有过采样发生的情况下 EVENTS_RESULTDONE 事件等同于 EVENTS_DONE 事件,注意在实际采样数据通过 EasyDMA 被保存到 RAM 之前,这两个事件都会产生。

1.3.3 连续转换

当我们使能一个 ADC 通道,ADC 工作于单次模式,当使能的通道数量大于 1 个,ADC进入扫描模式。
扫描模式下,采样所有通道花费的总时间如下式:

EVENTS_DONE 事件表明了一次采样的完成,扫描模式下,在没有过采样发生的情况下EVENTS_RESULTDONE 事件等同于 EVENTS_DONE 事件,注意在实际采样数据通过EasyDMA 被保存到 RAM 之前,这两个事件都会产生。

二、硬件连接

功能口 引脚
AIN0 P2
AIN1 P3
AIN2 P4
AIN3 P5
AIN4 P28
AIN5 P29
AIN6 P30
AIN7 P31

三、移植文件(单次采样)

注意:以下出现缺失common.h文件错误,去除即可。uint8改为uint8_t或unsigned char或自己宏定义
链接:https://pan.baidu.com/s/1rsa6nkGuU1QgEsQ0i1PGZQ 提取码:vpyt
board_adc.cboard_adc.h 两个文件加入工程的Application文件夹下


3.1board_adc.c

/*********************************************************************
 * INCLUDES
 */
#include "nrfx_saadc.h"
#include "nrf_drv_saadc.h"
#include "app_error.h" 

#include "board_adc.h"
#include "User/user_battery.h"

#include "nrf_log.h"

static void adcCallbackFunc(nrf_drv_saadc_evt_t const *pEvent);

/*********************************************************************
 * LOCAL VARIABLES
 */
static nrf_saadc_value_t s_bufferPool[SAMPLES_IN_BUFFER];
    
/*********************************************************************
 * PUBLIC FUNCTIONS
 */
/**
 @brief ADC的初始化函数
 @param 无
 @return 无
*/
void ADC_Init(void)
{
    ret_code_t errCode;
    // ADC初始化
    errCode = nrf_drv_saadc_init(NULL, adcCallbackFunc);
    APP_ERROR_CHECK(errCode);
    // ADC通道配置
    nrf_saadc_channel_config_t channelConfig = NRF_DRV_SAADC_DEFAULT_CHANNEL_CONFIG_SE(NRF_SAADC_INPUT_AIN0);   // 单端输入
    // ADC通道初始化
    errCode = nrf_drv_saadc_channel_init(0, &channelConfig);
    APP_ERROR_CHECK(errCode);
    // 缓冲配置
    errCode = nrf_drv_saadc_buffer_convert(s_bufferPool, SAMPLES_IN_BUFFER);
    APP_ERROR_CHECK(errCode);
}

/**
 @brief ADC读取
 @param 无
 @return 结果在回调函数的缓冲区中
*/
void ADC_Read(void)
{
    ret_code_t errCode;
    errCode = nrf_drv_saadc_sample();
    APP_ERROR_CHECK(errCode);
}

/**
 @brief 开启ADC,与初始化没有区别,为了与Disable成对出现
 @param 无
 @return 无
*/
void ADC_Enable(void)
{
    ADC_Init();
}

/**
 @brief 禁用ADC
 @param 无
 @return 无
*/
void ADC_Disable(void)
{
    nrfx_saadc_uninit();
}


/*********************************************************************
 * LOCAL FUNCTIONS
 */
/**
 @brief ADC中断处理回调函数
 @param 无
 @return 无
*/
static void adcCallbackFunc(nrf_drv_saadc_evt_t const *pEvent)
{
    if(pEvent->type == NRF_DRV_SAADC_EVT_DONE)                                                                  // 采样完成
    {
        nrf_saadc_value_t adcResult;
        uint16 batteryVoltage;
        uint8 batteryPercentage;
        ret_code_t errCode;
        
        // 设置好缓存,为下次转换缓冲做准备,并且把导入到缓冲的值提取出来
        errCode = nrf_drv_saadc_buffer_convert(pEvent->data.done.p_buffer, SAMPLES_IN_BUFFER);                  
        APP_ERROR_CHECK(errCode);
        adcResult = pEvent->data.done.p_buffer[0];
        // 电池电压转换计算
        batteryVoltage = ADC_RESULT_IN_MILLI_VOLTS(adcResult);
    }
}

/****************************************************END OF FILE****************************************************/

3.2 board_adc.h

#ifndef _BOARD_ADC_H_
#define _BOARD_ADC_H_

/*********************************************************************
 * INCLUDES
 */
#include "common.h"

/*********************************************************************
 * DEFINITIONS
 */
#define ADC_REF_VOLTAGE_IN_MILLIVOLTS   600                 /**< Reference voltage (in milli volts) used by ADC while doing conversion. */
#define ADC_PRE_SCALING_COMPENSATION    6                   /**< The ADC is configured to use VDD with 1/3 prescaling as input. And hence the result of conversion is to be multiplied by 3 to get the actual value of the battery voltage.*/
#define DIODE_FWD_VOLT_DROP_MILLIVOLTS  1000                    /**< Typical forward voltage drop of the diode . */
#define ADC_RES_10BIT                   1024                /**< Maximum digital value for 10-bit ADC conversion. */

// VP = (RESULT * REFERENCE / 2^10) * 6
#define ADC_RESULT_IN_MILLI_VOLTS(ADC_VALUE)\
        ((((ADC_VALUE) * ADC_REF_VOLTAGE_IN_MILLIVOLTS) / ADC_RES_10BIT) * ADC_PRE_SCALING_COMPENSATION) 
 
#define SAMPLES_IN_BUFFER               1

/*********************************************************************
 * API FUNCTIONS
 */
void ADC_Init(void);
void ADC_Read(void);
void ADC_Enable(void);
void ADC_Disable(void);

#endif /* _BOARD_ADC_H_ */

四、API调用

需包含头文件 board_adc.h

ADC_Init

功能 初始化ADC驱动
函数定义 void ADC_Init(void)
参数
返回

ADC_Read

功能 ADC读取函数
函数定义
参数
返回

ADC_Enable

功能 使能ADC,内部重新初始化ADC驱动,在ADC_Disable之后使用
函数定义 void ADC_Enable(void)
参数
返回

ADC_Disable

功能 禁用ADC,以进入低功耗
函数定义 void ADC_Disable(void)
参数
返回

五、SDK配置

点击 sdk_config.h 文件


选择 Configuration Wizard

nRF_Drivers 中勾选SAADC相关选项

六、使用例子

1)添加头文件

#include "board_adc.h"

2)添加初始化代码(SDK15.3 中 ble_peripheral 的 ble_app_template 工程 main() 函数中)
加入 ADC_Init() 并在初始化后调用 ADC_Disable 进入低功耗,在需要用ADC时调用 ADC_Enable 开启ADC

int main(void)
{
    bool erase_bonds;

    /*-------------------------- 外设驱动初始化 ---------------------------*/
    // Initialize.
    log_init();                                                                 // 日志驱动初始化                                                                  
    timers_init();                                                              // 定时器驱动初始化(在此加入自定义定时器)
    ADC_Init();                                                                 // ADC驱动初始化 
    
    /*-------------------------- 蓝牙协议栈初始化 ---------------------------*/
    power_management_init();
    ble_stack_init();                                                           // 协议栈初始化
    gap_params_init();
    gatt_init();
    advertising_init();                                                         // 广播初始化
    services_init();                                                            // 服务初始化
    conn_params_init();                                                         // 连接参数初始化
    peer_manager_init();
    
    /*-------------------------- 开启应用 ---------------------------*/
    // Start execution.
    NRF_LOG_INFO("Template example started."); 
    advertising_start(erase_bonds);                                             // 开启广播 
    application_timers_start();                                                 // 定时器应用开启(在此开启自定义定时器)  
    ADC_Disable();                                                              // 禁用ADC,进入低功耗模式,等待读取时再开启
    
    // Enter main loop.
    for(;;)
    {
        idle_state_handle();
    }
}

3)在定时器回调中开启ADC读取,然后再开一个10ms定时器关闭ADC进入低功耗

static void timer_batteryMeasureOnCallback(void *arg)
{
    UNUSED_PARAMETER(arg);
    ADC_Enable();                                                                           // 开启ADC
    ADC_Read();
    app_timer_start(s_batteryMeasureOffTimer, BATTERY_MEASURE_OFF_PERIOD, NULL);    
}

4)在ADC中断处理回调函数中,将采集ADC值进行电压值转换

/**
 @brief ADC中断处理回调函数
 @param 无
 @return 无
*/
static void adcCallbackFunc(nrf_drv_saadc_evt_t const *pEvent)
{
    if(pEvent->type == NRF_DRV_SAADC_EVT_DONE)                                                                  // 采样完成
    {
        nrf_saadc_value_t adcResult;
        uint16 batteryVoltage;
        ret_code_t errCode;
        
        // 设置好缓存,为下次转换缓冲做准备,并且把导入到缓冲的值提取出来
        errCode = nrf_drv_saadc_buffer_convert(pEvent->data.done.p_buffer, SAMPLES_IN_BUFFER);                  
        APP_ERROR_CHECK(errCode);
        adcResult = pEvent->data.done.p_buffer[0];
        // 电池电压转换计算
        batteryVoltage = ADC_RESULT_IN_MILLI_VOLTS(adcResult);
    }
}

• 由 Leung 写于 2020 年 1 月 6 日

• 参考:青风电子社区
    Nordic--nrf52832--ADC

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352