⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ad.c

📁 AVR单片机ATmega16的AD用AVRISP板实验成功的例子
💻 C
字号:
#include <avr/io.h>#include <avr/delay.h>#include <avr/signal.h>#include <avr/interrupt.h>/*宏INTERRUPT 的用法与SIGNAL 类似,区别在于    SIGNAL 执行时全局中断触发位被清除、其他中断被禁止    INTERRUPT 执行时全局中断触发位被置位、其他中断可嵌套执另外avr-libc 提供两个API 函数用于置位和清零全局中断触发位,它们是经常用到的。分别是:void sei(void) 和void cli(void) 由interrupt.h定义 */
//管脚定义
#define in_Single 0 //PA0(ADC0)
#define in_Diff_P 3 //PA3(ADC3)
#define in_Diff_N 2 //PA2(ADC2)

//常量定义
//单端通道,不放大
#define AD_SE_ADC0 0x00 //ADC0
#define AD_SE_ADC1 0x01 //ADC1
#define AD_SE_ADC2 0x02 //ADC2
#define AD_SE_ADC3 0x03 //ADC3
#define AD_SE_ADC4 0x04 //ADC4
#define AD_SE_ADC5 0x05 //ADC5
#define AD_SE_ADC6 0x06 //ADC6
#define AD_SE_ADC7 0x07 //ADC7

//差分通道ADC0作负端,10/200倍放大
#define AD_Diff0_0_10x 0x08 //ADC0+ ADC0-, 10倍放大,校准用
#define AD_Diff1_0_10x 0x09 //ADC1+ ADC0-, 10倍放大
#define AD_Diff0_0_200x 0x0A //ADC0+ ADC0-,200倍放大,校准用
#define AD_Diff1_0_200x 0x0B //ADC1+ ADC0-,200倍放大

//差分通道ADC2作负端,10/200倍放大
#define AD_Diff2_2_10x 0x0C //ADC2+ ADC2-, 10倍放大,校准用
#define AD_Diff3_2_10x 0x0D //ADC3+ ADC2-, 10倍放大
#define AD_Diff2_2_200x 0x0E //ADC2+ ADC2-,200倍放大,校准用
#define AD_Diff3_2_200x 0x0F //ADC3+ ADC2-,200倍放大

//差分通道ADC1作负端,不放大
#define AD_Diff0_1_1x 0x10 //ADC0+ ADC1-
#define AD_Diff1_1_1x 0x11 //ADC1+ ADC1-,校准用
#define AD_Diff2_1_1x 0x12 //ADC2+ ADC1-
#define AD_Diff3_1_1x 0x13 //ADC3+ ADC1-
#define AD_Diff4_1_1x 0x14 //ADC4+ ADC1-
#define AD_Diff5_1_1x 0x15 //ADC5+ ADC1-
#define AD_Diff6_1_1x 0x16 //ADC6+ ADC1-
#define AD_Diff7_1_1x 0x17 //ADC7+ ADC1-

//差分通道ADC2作负端,不放大
#define AD_Diff0_2_1x 0x18 //ADC0+ ADC2-
#define AD_Diff1_2_1x 0x19 //ADC1+ ADC2-
#define AD_Diff2_2_1x 0x1A //ADC2+ ADC2-,校准用
#define AD_Diff3_2_1x 0x1B //ADC3+ ADC2-
#define AD_Diff4_2_1x 0x1C //ADC4+ ADC2-
#define AD_Diff5_2_1x 0x1D //ADC5+ ADC2-

//单端通道,不放大
#define AD_SE_VBG 0x1E //VBG 内部能隙1.22V电压基准,校准用
#define AD_SE_GND 0x1F //接地 校准用

//注:
//差分通道,如果使用1x或10x增益,可得到8位分辨率。如果使用200x增益,可得到7位分辨率。
//在PDIP封装下的差分输入通道器件未经测试。只保证器件在TQFP与MLF封装下正常工作。

#define Vref 2556 //mV 实测的Vref引脚电压@5.0V供电
//#define Vref 2550 //mV 实测的Vref引脚电压@3.3V供电

//全局变量
unsigned int ADC_SingleEnded; //单端输入的ADC值
int ADC_Diff; //差分输入的ADC值
volatile unsigned int ADC_INT_SE; //中断模式用的单端输入ADC值,会在中断服务程序中被修改,
                                  //须加volatile限定
volatile unsigned char ADC_OK; //ADC状态,会在中断服务程序中被修改,须加volatile限定
unsigned int LED_Volt; //变换后的电压mV
int LED_Curr; //变换

//仿真时在watch窗口,监控这些全局变量。

unsigned int read_adc(unsigned char adc_input)//查询方式读取ADC单端通道
   {
      ADMUX=(0xc0|adc_input); //adc_input:单端通道 0x00~0x07,0x1E,0x1F
      //0xc0:选择内部2.56V参考电压
      ADCSRA|=(1<<ADSC); //启动AD转换
      loop_until_bit_is_set(ADCSRA,ADIF); //方法1 等待AD转换结束
      // while ((ADCSRA&(1<<ADIF))==0); //写法2 这种写法优化不好
      // loop_until_bit_is_clear(ADCSRA,ADSC); //方法2 检测ADSC=0也行
      ADCSRA|=(1<<ADIF); //写1清除标志位
      return ADC; //ADC=ADCH:ADCL
   }
int read_adc_diff(unsigned char adc_input)//查询方式读取ADC差分通道
  {
      unsigned int ADC_FIX;
      ADMUX=(0xc0|adc_input); //adc_input:差分通道 0x08~0x1D
      _delay_ms(1); //等待差分增益稳定>125uS
      ADCSRA|=(1<<ADSC);
      loop_until_bit_is_set(ADCSRA,ADIF);
      ADCSRA|=(1<<ADIF);
      //当切换到差分增益通道,由于自动偏移抵消电路需要沉积时间,
      //第一次转换结果准确率很低。
      //用户最好舍弃第一次转换结果。
      ADCSRA|=(1<<ADSC);
      loop_until_bit_is_set(ADCSRA,ADIF);
      ADCSRA|=(1<<ADIF);
      ADC_FIX=ADC;
      //输出结果用2的补码形式表示
      //可正可负 +/-9bit -512~+511
      //即M16差分通道的ADC+输入端的电压可以大于ADC-,也可以小于ADC-。
      //Tiny26就不行,ADC+输入端的电压必须大于或等于ADC-,为+10bit
   
     if (ADC_FIX>=0x0200) //负数要变换,正数不用
        {
           ADC_FIX|=0xFC00; //变换成16位无符号整数
         }
     return (int)ADC_FIX;
    }

SIGNAL(SIG_ADC) //ADC中断服务程序
   {
      //硬件自动清除ADIF标志位
      ADC_INT_SE=ADC; //读取结果
      ADC_OK=1;
   }

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -