1、参考电压。参考电压用来与输入的模拟量进行比较,作为测量的基准。一般REF(=)5vREF(-)0V。下面我先给出ADC0809的时序图再说说它的工作过程:它的工作过程是这样的,在IN0IN7上可分别接上要测量转换的8路模拟量信号。有人问了,可不可以只接一路?我就只想测一个模拟信号。当然可了,能挑一百斤的担子,让你只挑十斤那还不小菜。废话太多。STOP。将ADDAADDC端给上代表选择测量通道的代码。如000(B)则代表通道0;001(B)代表通道1;111则代表通道7。将ALE由低电平置为高电平,从而将ADDAADDC送进的通道代码锁存,经译码后被选中的通道的模拟量送给内部转换单元。给STAR
2、T一个正脉冲。当上升沿时,所有内部寄存器清零。下降沿时,开始进行A/D转换;在转换期间,START保持低电平。EOC为转换结束信号。在上述的A/D转换期间,可以对EOC进行不断测量,当EOC为高电平时,表明转换工作结束。否则,表明正在进行A/D转换。当A/D转换结束后,将OE设置为1,这时D0D7的数据便可以读取了。OE0,D0D7输出端为高阻态,OE1,D0D7端输出转换的数据。说明:ADC0809的转换工作是在时钟脉冲的条件下完成的,因此首先要在CLOCK端给它一个时钟信号,说明书上给出了可以接入的脉冲信号频率是在10KHz1280KHz,典型值是640KHz。时序图上的teoc时长为,从
3、START上升沿开始后的8个时钟同期再加2微秒。这一点得注意,因为当START脉冲刚结束进入转换工作时,EOC还没有立即变为低电平而是过了8个时钟周期后才进入低电平的,所以再给出START脉冲后最好延时一会再进行EOC的检测。一个通道的转换时间一般为64个时钟周期,如时钟频率为640KHz时,时钟周期为1.5625微秒,一个通道的转换时间则为1.562564100微秒,那么1秒种就可以转换100000010010000次。下面我们给出一个教材上经典的接线图再说明它的利弊:这是个老教材上的图纸,网络查查也大都是这样的图,它的时钟脉冲是通过C51的ALE经过2分频得到的,它一般是指,当C51的晶振
4、为6MHz时,ALE输出1MHz的脉冲,经2分频后得到500KHz。但这对我们现在S52上使用12MHz晶振时就不适用了。现在S52的ALE输出的脉冲为2MHz,2分频后也有1MHz。网上查了一下,补救的办法基本上是用T0时钟来模拟出个10KHz的脉冲信号。这是ADC0809时钟脉冲允许的最小值。也的确,一般我们用不着那么快的转换速度,所以给个10KHz也够了,还省掉了一个分频器。不过这也基本上是用T0能模拟出的最高频率了,sbit CLK=P33;void main(void)ET01;EA=1;TMOD=0X12;TH0=216;TL0=216;TR0=1;.void t0(void) i
5、nterrupt 1CLK=CLK;从上面的程序可以看出,每40个时钟脉冲就发生一次中断。S51基本上也就一直忙着中断处理了。中断太频繁,占资源。我们来看看D0D7输出口,它是只能输出不能写入的,而ADDAADDC又是只能写入而不能输出的,因此我们可以将74HC373也省掉。同样74LS02也自然可以不用了。这样我们省去了所有的其它门电路IC,也包括那个HC373。简化电路为的是使ADC0809在用最少的其它门电路而使它工作,让我们能专注学习ADC0809是如何工作过程的。现在我来说说如何更好的模拟出这个时钟。我们现在用的都是S52芯片了,不知道大家用过里面的T2时钟吗?它有个时钟输出功能,如
6、下图:根据公式,用12MHz的晶振,就可以在P1.0输出45Hz-3MHz频率的时钟脉冲,要输出500KHz的脉冲还不小意思呀。我试验了一下T2时钟,很好用:/*这个程序让AT89S52的T2时钟从P1.0口输出45HZ方波脉冲,P1.0口接到P3.4口作为T0的外部计数,通过T0的时钟中断在P2口输出。T0每45个脉冲就中断一次,也就是刚好一秒一次*/#include void main()/* T2 设置 */TR2=0x0;T2MOD=0x02; /0010(B) 设置T2为P1.0口输出方波模式C_T2=0; /用内部时钟计数TL2=0x0;TH2=0x0;RCAP2L=0x0;RCA
7、P2H=0x0;/* T0 设置 */TMOD=0x6; /0110(B),T0为外部计数模式,方式为2 (8位自动装载)/总中断允许ET0=1;/T0中断允许TH0=256-45;TL0=256-45;/*start timer */ /启动T0时钟TR2=1; /启动T2时钟do while(1);void t0(void) interrupt 1/T0中断服务程序 P2=P2;其结果P1.0输出45Hz的脉冲,T0的中断是以P1.0的45个脉冲发生一次的。P2口的数码管以一秒的时间亮,一秒时间熄灭的在闪烁,T2定时器看来很好用。达到要求。好!上面的程序只是试试S52的T2定时器是否可正常
8、使用。言归正传我们还是来谈ADC0809的电路:见下图:电路图比较简单,我就不另画了,就将这张电图板图贴上。D0D7是数据读取位,同时在低三位D0D2上也分别接上了A、B、C通道选择位。CLK为ADC0809所需的时钟脉冲,我们将要用T2来给出时钟脉冲,因此我们必须把CLK接在S52的P1.0上。这样OE、EOC、ST就分别顺着接P1.1、P1.2、P1.3。上面除了ADC0809,就接了个电源指示用的LED和两个电源滤波电容。下面是做好后的实物图:下一部分我们讲实际程序和具体调试。一篇我们已经讲了ADC0809的原理与简化电路,仔细检查电路的正确性,尤其不能有短路现象。现在我们将它实际接上S
9、52板子,静态电流为4mA,其中LED用了1mA,也就是说ADC0809静态电流为3mA左右。将ADC0809板插上S52板,数据口D0D7我接的是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程序和汇编程序一块儿编译。我们先来看主程序:#define
10、uchar unsigned charextern 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 str4; /定义四个字节的数组,用来存放将数值转成的字符/*/将char值转成字符串函数uch
11、ar * 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) strn=x00+48; /值加48即为字符 n+;if(!(x00=0&x0=0) strn=x0+48;strn=x+48;n+;strn=0; return str;/*/ 主函数void Main( void ) uchar aa; /定义一个临时字符变量 /* T2 set */ /停止T2定时器TR0=0x0; /停止T0定时器TL2=0xfd;TH2=
12、0xff;RCAP2L=0xfd;RCAP2H=0xff;TMOD=0x01; /设置T0为1定时模式(16位计数)TH0=0;TL0=0; /打开T2定时器,开始输出脉冲aa=adc0809conv(); /启动一次ADC0809转换并将值交给aaLcmInit(); /初始化LCDLcmClear(); /LCD清屏LcmPutstr( 2,28,ADC0809 TEST );LcmPutstr( 4,59,uchartostr(aa) ); /在第四行第59列输出ADC0809转换的值LcmPutstr( 7,42,TXZ001while(1) 下面我们再来看ADC0809转换函数:NA
13、ME ADC0809C?PR?adc0809conv?ADC0809C SEGMENT CODE DT? SEGMENT DATA OVERLAYABLE PUBLIC adc0809convRSEG ?ADC0809CBYTE: put?040: DS 1adc0809conv: ;程序从这里开始USING 0st bit P1.3设置ST接P1.3eoc bit P1.2设置EOC接P1.2oe bit P1.1设置OE接P1.1port equ P0设置数据读取PORT接P0setb TR0启动T0定时器,用来计数(我是用T0来计算转换一次需要多长时间)clr oe初始化ADC0809,
14、OE置0clr st初始化ADC0809,ST置0setb eoc初始化ADC0809,EOC置1mov port,#0先择通道0数据交给P0口setb st这三句将ST给出一个正脉冲来启动转换nopmov 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,po
15、rt将转换的数据交给主调用程序的变量aaC0001:RET ;返回END汇编程序有点乱,没关系,下一篇我会专门讲混合编程。那个LCD显示函数就不在这儿列出了,前面都已讲过也列出了程序。下图是本程序实际测量一节AA电池的实例图:可以看到,它的值为64,因为我的ADC0809参考电压为5V,那么8位精度的转换是将5v分为255份那么每份就是52550.0196v。我测量出一节电池的64值就为640.01961.25v。调试注意点:由于8个模拟测量通道的输入阻抗很高,在程序运行时如果8个模拟端是悬空的,模拟端的电位是随周围环境变化的,那测量出的很可能是乱跳的随机值而并非你程序或电路问题。要避免这种情
16、况的干扰,最好先用10K的电阻将模拟端接地。等你测量出每次都为0时,再改变模拟端的电位试验测量的正确性。随便说说转换速度的问题。ADC0809的转换速度跟脉冲频率有关,它的允许范围为10KHZ1.28M,我们是用T2定时器来做脉冲输出的,频率就由公式晶振频率/(4(65536(RCAP2H,RCAP2L),还记得上一篇我给出过的这个公式吗?我的晶振是12MHZ,那么要给出1MHZ的脉冲就要在RCAP2里给65533的值,这样12M431MHZ。同理,要输出10KHZ的脉冲就要给65236的值。在主程序里:TL2=0xfd这是计数器里的初值,FFFD就是65533,也就是输出1MHZ的脉冲TH2
17、=0xffRCAP2L=0xfd这是重载器,也一样给上65533的值。RCAP2H=0xff如果要想输出10KHZ的脉冲上面就要给上65236的值,也就是FED4。在主程序里我还用了T0计数器:TMOD=0x01;TH0=0;TL0=0;我给的初值为0,但我没有在主程序里启动它。而是在ADC0809转换函数里才启动和停止:setb TR0。clr TR0这样我就可以看它计数的值来知道一次转换需要多长时间了。将转换函数的最后一句: 换成MOV R7,TL0或MOV R7,TH0 就会在LCD上显示出转换所用的时间了,因为晶振为12MHZ,一个脉冲就是一微秒。实际测试ADC0809在1MHZ时钟时转换一次为83微秒,而在10KHZ时钟下转换一次需要1769微秒也就是1.769毫秒,比ADC0804转换速度要慢很多,ADC0804为22微秒。
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1