黑塔利亚普爷:模数转换IC ADC0809的使用制做之二(转载)

来源:百度文库 编辑:九乡新闻网 时间:2024/05/02 12:40:26
上一篇我们已经讲了ADC0809的原理与简化电路,仔细检查电路的正确性,尤其不能有短路现象。现在我们将它实际接上S52板子,静态电流为4mA,其中LED用了1mA,也就是说ADC0809静态电流为3mA左右。



将ADC0809板插上S52板,数据口D0-D7我接的是P0,CLK接P1.0,OE、EOC、ST-ALE分别接P1.1、P1.2、P1.3,Vcc和GND接S52板上的电源和地。

下面讲程序:
程序分为三个部分:主程序、LCD显示、ADC0809转换。

左边的项目框里有三个程序文件:主程序ADC0809m.C、LCD显示程序12864put.c、ADC0809转换程序ADC0809C.asm。嗯?。。。最后一个文件是汇编文件?是的!KEIL允许同时加入C程序和汇编程序一块儿编译。我们先来看主程序:
#include
#define uchar unsigned char
extern void LcmClear( void );       //清屏,外部函数
extern void LcmInit( void );        //初始化,外部函数
extern void LcmPutstr( uchar row,uchar y,uchar * str ); //在设定位置显示字符串
extern uchar adc0809conv(void); //
uchar * uchartostr(unsigned char unm);    //将char值转成字符串
uchar str[4];    //定义四个字节的数组,用来存放将数值转成的字符

//****************************
//将char值转成字符串函数
//****************************
uchar * uchartostr(uchar unm)
{
uchar x00,xx,x0,x,n; //定义百位,十位,个位变量
x00=unm/100;
xx=unm%100;
x0=xx/10;
x=xx%10;
n=0;
if(x00!=0)
{ str[n]=x00+48; //值加48即为字符
    n++;
}
if(!(x00==0&x0==0))
{ str[n]=x0+48;
    n++;
}
str[n]=x+48;
n++;
str[n]='\0';
   return str;
}

//****************
//    主函数
//****************

void Main( void )
{ uchar aa;    //定义一个临时字符变量

    /* T2 set */
TR2=0x0;     //停止T2定时器
TR0=0x0;     //停止T0定时器
T2MOD=0x02; //0010(B) 设置T2为P1.0口输出方波模式
C_T2=0;      //用内部时钟计数
TL2=0xfd;
TH2=0xff;
RCAP2L=0xfd;
RCAP2H=0xff;
TMOD=0x01; //设置T0为1定时模式(16位计数)
TH0=0;
TL0=0;
TR2=1;    //打开T2定时器,开始输出脉冲
aa=adc0809conv(); //启动一次ADC0809转换并将值交给aa
LcmInit();         //初始化LCD
LcmClear();        //LCD清屏
LcmPutstr( 2,28,"ADC0809 TEST" );
LcmPutstr( 4,59,uchartostr(aa) ); //在第四行第59列输出ADC0809转换的值
LcmPutstr( 7,42,"TXZ001@139.com" );
while(1)
    {
      
    }
}

下面我们再来看ADC0809转换函数:

NAME ADC0809C
?PR?adc0809conv?ADC0809C                 SEGMENT CODE
?DT?adc0809conv?ADC0809C                 SEGMENT DATA OVERLAYABLE
PUBLIC adc0809conv
RSEG ?DT?adc0809conv?ADC0809C
?adc0809conv?BYTE:
        put?040:   DS   1
RSEG ?PR?adc0809conv?ADC0809C
adc0809conv:            ;程序从这里开始
USING 0
st bit P1.3     ;设置ST接P1.3
eoc bit P1.2    ;设置EOC接P1.2
oe bit P1.1     ;设置OE接P1.1
port equ P0     ;设置数据读取PORT接P0
setb TR0        ;启动T0定时器,用来计数(我是用T0来计算转换一次需要多长时间)
clr oe          ;初始化ADC0809,OE置0
clr st          ;初始化ADC0809,ST置0
setb eoc        ;初始化ADC0809,EOC置1
mov port,#0     ;先择通道0数据交给P0口
setb st         ;这三句将ST给出一个正脉冲来启动转换
nop       ;
clr st          ;
mov r7,#10      ;这两行是用来稍做延时
djnz r7,$       ;
wait1: jb eoc,wait1   ;这两行是来检测EOC由低到高发出了上升沿,以表示转换结束
wait2: jnb eoc,wait2 ;
   mov port,#0FFh       ;将P0口复位,以便下一步读取数据
setb oe         ;将OE口置1,允许转换后的数据读出
clr TR0         ;停止T0计时器(T0是从0开始计数的,到这儿转换结束停止计数)
MOV R7,port    ;将转换的数据交给主调用程序的变量aa

?C0001:
RET ;返回
END

汇编程序有点乱,没关系,下一篇我会专门讲混合编程。
那个LCD显示函数就不在这儿列出了,前面都已讲过也列出了程序。
下图是本程序实际测量一节AA电池的实例图:


可以看到,它的值为64,因为我的ADC0809参考电压为5V,那么8位精度的转换是将5v分为255份那么每份就是5÷255=0.0196v。我测量出一节电池的64值就为64×0.0196=1.25v。

调试注意点:由于8个模拟测量通道的输入阻抗很高,在程序运行时如果8个模拟端是悬空的,模拟端的电位是随周围环境变化的,那测量出的很可能是乱跳的随机值而并非你程序或电路问题。要避免这种情况的干扰,最好先用10K的电阻将模拟端接地。等你测量出每次都为0时,再改变模拟端的电位试验测量的正确性。

随便说说转换速度的问题。
  ADC0809的转换速度跟脉冲频率有关,它的允许范围为10KHZ--1.28M,我们是用T2定时器来做脉冲输出的,频率就由公式 晶振频率/(4×(65536-(RCAP2H,RCAP2L)),还记得上一篇我给出过的这个公式吗?我的晶振是12MHZ,那么要给出1MHZ的脉冲就要在RCAP2里给65533的值,这样12M÷4×3=1MHZ。同理,要输出10KHZ的脉冲就要给65236的值。在主程序里:

  TL2=0xfd    ;这是计数器里的初值,FFFD就是65533,也就是输出1MHZ的脉冲
  TH2=0xff    ;
  RCAP2L=0xfd    ;这是重载器,也一样给上65533的值。
  RCAP2H=0xff    ;

如果要想输出10KHZ的脉冲上面就要给上65236的值,也就是FED4。

在主程序里我还用了T0计数器:
  TMOD=0x01; //设置T0为1定时模式(16位计数)
  TH0=0;
  TL0=0;
我给的初值为0,但我没有在主程序里启动它。而是在ADC0809转换函数里才启动和停止:

   setb TR0        ;启动T0定时器,用来计数(我是用T0来计算转换一次需要多长时间)
   。。。。
   clr TR0         ;停止T0计时器(T0是从0开始计数的,到这儿转换结束停止计数)

这样我就可以看它计数的值来知道一次转换需要多长时间了。将转换函数的最后一句:
MOV R7,port    换成 MOV R7,TL0 或 MOV R7,TH0 就会在LCD上显示出转换所用的时间了,因为晶振为12MHZ,一个脉冲就是一微秒。实际测试ADC0809在1MHZ时钟时转换一次为83微秒,而在10KHZ时钟下转换一次需要1769微秒也就是1.769毫秒,比ADC0804转换速度要慢很多,ADC0804为22微秒。

怎么样,单片机好玩吧!