现在的位置: 首页 > 综合 > 正文

zigbee cc2530 adc转换

2013年10月12日 ⁄ 综合 ⁄ 共 3133字 ⁄ 字号 评论关闭

cc2530的通用datasheet上没怎么讲到adc的一些特性,

http://download.csdn.net/detail/songqqnew/5132088

而是下面这个文档有较多讲解
www.ti.com/lit/ds/symlink/cc2530.pdf

#include "adc.h"
void main(void)
{
  while(1)
     adcSampleSingle (ADC_REF_AVDD,ADC_12_BIT,ADC_AIN0);
}

1.源于basic代码
2.cc2530没有专门设置分辨率的寄存器,有一个抓取率,设置这个东东就相当于设置了分辨率和adc转换时间,(套餐,分辨率越高,转换时间越长)
单个测量时,分辨率最高为12位,达不到14位,
在抓取率是512时可以达到最大分辨率(同时也达到最慢的转换速率)。
datasheeet上如此介绍
The ADC supports up to 14-bit analog-to-digital conversion with up to 12 bits ENOB (Effective Number OfBits).
最高支持14位ad转换分辨率,但是有效位是12位。看看是那几位:
ADCH的最高位是符号位,对于单个测量,结果总是正,所以符号位总是0。所以用到bit6--bit0,计7位
ADCL的低2位(bit0,bit1)系统保留,bit2不用,所以用到bit7--bit3,计5位。
所以共12位。
当然其他抓取率时,有效分辨率如下:
00: 64 decimation rate (7 bits ENOB)--ADCH低7位
01: 128 decimation rate (9 bits ENOB)--ADCH低7位+ADCH高2位
10: 256 decimation rate (10 bits ENOB)--ADCH低7位+ADCH高3位
11: 512 decimation rate (12 bits ENOB)--ADCH低7位+ADCL高5位

在分辨率是7位时,如果按照分辨率是12去取结果,是不太准确的。

3.参考电压
#define ADC_REF_1_25_V      0x00     // Internal 1.25V reference
#define ADC_REF_P0_7        0x40     // External reference on AIN7 pin
#define ADC_REF_AVDD        0x80     // AVDD_SOC pin,=3.3v
#define ADC_REF_P0_6_P0_7   0xC0     // External reference on AIN6-AIN7 differential input

其实如果连符号位和bit2也算上,最高分辨率是14位。

不过不如协议栈的adc转换函数更加可信:

直接使用即可
#include "hal_adc.h"
uint16 u16cvalu=HalAdcRead(HAL_ADC_CHANNEL_4,HAL_ADC_RESOLUTION_12);
分辨率设置为12位时,从源码可以看出,可用位是ADCH 8位+ADCH高4位,其中ADCH最高位是符号位,所以有11位的分辨率,0-2047
默认基准电压3.3V

uint16 HalAdcRead (uint8 channel, uint8 resolution)
{
  int16  reading = 0;

#if (HAL_ADC == TRUE)

  uint8   i, resbits;
  uint8   adctemp;
  volatile  uint8 tmp;
  uint8  adcChannel = 1;
  uint8  reference;

  /* store the previously set reference voltage selection */
  reference = ADCCON3 & HAL_ADC_REF_BITS;

  /*
  * If Analog input channel is AIN0..AIN7, make sure corresponing P0 I/O pin is enabled.  The code
  * does NOT disable the pin at the end of this function.  I think it is better to leave the pin
  * enabled because the results will be more accurate.  Because of the inherent capacitance on the
  * pin, it takes time for the voltage on the pin to charge up to its steady-state level.  If
  * HalAdcRead() has to turn on the pin for every conversion, the results may show a lower voltage
  * than actuality because the pin did not have time to fully charge.
  */
  if (channel < 8)
  {
    for (i=0; i < channel; i++)
    {
      adcChannel <<= 1;
    }
  }

  /* Enable channel */
  ADCCFG |= adcChannel;

  /* Convert resolution to decimation rate */
  switch (resolution)
  {
    case HAL_ADC_RESOLUTION_8:
      resbits = HAL_ADC_DEC_064;
      break;
    case HAL_ADC_RESOLUTION_10:
      resbits = HAL_ADC_DEC_128;
      break;
    case HAL_ADC_RESOLUTION_12:
      resbits = HAL_ADC_DEC_256;
      break;
    case HAL_ADC_RESOLUTION_14:
    default:
      resbits = HAL_ADC_DEC_512;
      break;
  }

  /* read ADCL,ADCH to clear EOC */
  tmp = ADCL;
  tmp = ADCH;

  /* Setup Sample */
  adctemp = ADCCON3;
  adctemp &= ~(HAL_ADC_CHN_BITS | HAL_ADC_DEC_BITS | HAL_ADC_REF_BITS);
  adctemp |= channel | resbits | (reference);

  /* writing to this register starts the extra conversion */
  ADCCON3 = adctemp;

  /* Wait for the conversion to be done */
  while (!(ADCCON1 & HAL_ADC_EOC));

  /* Disable channel after done conversion */
  ADCCFG &= (adcChannel ^ 0xFF);

  /* Read the result */
  reading = (int16) (ADCL);
  reading |= (int16) (ADCH << 8);

  /* Treat small negative as 0 */
  if (reading < 0)
    reading = 0;

  switch (resolution)
  {
    case HAL_ADC_RESOLUTION_8:
      reading >>= 8;
      break;
    case HAL_ADC_RESOLUTION_10:
      reading >>= 6;
      break;
    case HAL_ADC_RESOLUTION_12:
      reading >>= 4;
      break;
    case HAL_ADC_RESOLUTION_14:
    default:
      reading >>= 2;
    break;
  }
#else
  // unused arguments
  (void) channel;
  (void) resolution;
#endif

  return ((uint16)reading);
}

抱歉!评论已关闭.