SPEED0 BIT P1.0 SPEED1 BIT P1.1 SPEED2 BIT P1.2 SPEED3 BIT P1.3 KEYDOOROPEN BIT P1.4 KEYDOORCLOSE BIT P1.5 MODE BIT P1.6 DOOROPEN BIT P3.0 DOORCLOSE BIT P3.1 SIGNAL BIT P3.2 ADC0809EOC BIT P3.3 HSPEED BIT P3.4 DOORSTATE BIT 00H RERROR BIT 01H ADDR0809 BIT 60H ORG 0000H AJMP START ORG 0003H AJMP INT0 ORG 000BH AJMP T0 ORG 001BH AJMP T1 ORG 0050H
START: CLR DOOROPEN ;关电机驱动 CLR DOORCLOSE
MOV TMOD, #11H ;定时器工作方式
MOV TL0,#0F0H ;置定时器常数,T01s,T12s, 12MHz MOV TH0,#08DH MOV TL1,#0E0H MOV TH1,#0B1H MOV TCON,#50H
MOV IP,#08H ; 优先级设置 MOV IE,#80H ; 开中断
LOOP: ACALL SYSERROR ; 调用系统故障子程序 JB RERROR,WAIT ; 有故障等待处理 MOV A,P2 ANL A,#0F0H
WAIT: JNZ WAIT ; 无按键等待 JB MODE,AUTO ; 运行方式判断 CLR EX0 SETB ET0 SJMP LOOP AUTO: SETB ET0
SJMP LOOP
DOOR_OPEN: JB DOORSTATE, LOOP3 ;门已开退出 CLR HSPEED
CLR DOORCLOSE
SETB DOOROPEN ;低速启动 JNB SPEED1, $
SETB HSPEED ;高速开门 JNB SPEED2,$
CLR HSPEED ;低速运行 JNB SPEED3,$
CLR DOOROPEN ;停机
SETB DOORSTATE ;保存门状态 LOOP3: RET ;关门子程序 DOOR_CLOSE:
JNB DOORSTATE, LOOP4 ;门已关退出
CLR HSPEED CLR DOOROPEN
SETB DOORCLOSE ;低速启动 JB SPEED2,$
CLR HSPEED ;高速关门 JB SPEED0,$
CLR DOORCLOSE ;停机 CLR DOORSTATE LOOP4: RET ;T0中断服务程序:
T0: CLR ET0 ;关中断 JB KEYDOOROPEN,LOOP1 ;开门否
ACALL DOOR_OPEN ;调开门子程序 SJMP LOOP2
LOOP1: JB KEYDOORCLOSE, LOOP2 ; 关门否
ACALL DOOR_OPEN ; 调关门子程序 LOOP2: MOV TL0, #0F0H ;重装定时常数 MOV TH0, #08DH
SETB ET0 ;开中断 RETI
;T1中断服务程序:
MAIN: ORL P1,#0FFH
SETB T0 ;选择边沿触发方式 SETB EX0 ;允许INTO中断 SETB EA ; CPU开中断 AJMP $ ;等待中断 T1: CLR ET1
MOV R1,#03H
MOV R0,#20H
MOV DPTR,#00H ;0809地址 MOV R2,#00H
LOOP5:MOV A,R2 ;通道号
MOVX @DPTR,A ;启动A/D转换 JB ADC0809EOC,$ ;转换是否结束 MOVX A,@DPTR ;读取转换结果 MOV @R0,A ;数据保存 INC R0
INC R2 DJNZ R1,LOOP5 MOV TL1,#0E0H MOV TH1,#0B1H SETB ET1 RETI
;外部中断服务子程序: ORG 0300H INT: MOV TL0,#11H MOV TH0,#0FCH CPL P1 RETI
INT0: CLR EX0
LOOP8: ACALL SYSERROR JB RERROR,LOOP6 JB DOORSTATE,LOOP6 ACALL DOOR_OPEN LOOP7: ACALL DELAY10S ACALL SYSERROR JB RERROR,LOOP6 JNB SIGNAL,LOOP7 CLR HSPEED
SETB DOORCLOSE JNB SIGNAL,LOOP8 JB SPEED2,$
SETB HSPEED JNB SIGNAL,LOOP8 JB SPEED1,$
CLR HSPEED JB SPEED0,$
LOOP6: CLR DOORCLOSE CLR DOORSTATE SETB EX1 RET
DELAY10S:MOV R5,#255
; 下一通道
;下一个通道是否完成 ;重装定时常数 ;故障检查 ;有故障返回 ;门已开返回 ;开门 ;等待10s ;故障检查 ;无故障继续 ;有人等待 ;无人启动电机,低速关门 ;有人来打开门 ;无人快速关门 ;有人来打开门 ;无人速度降低 ;停机
;门状态保存 AA: MOV R6,#0FFH DJNZ R6,$ DJNZ R5,AA RET
SYSERROR: SJMP $ RET END
ADC0809
2011-08-08 14:57
ADC0809是单片机教材上常常用到的模数转换芯片,它有8个模拟输入通道,每次可选其中一路,转换成8位二进制数。
在PROTEUS仿真环境中,ADC0809并没有仿真功能(也许是版本的问题),仿真必须使用ADC0808。
这两个芯片在PROTEUS中的图形可见下图:
从图中可以看到,这两块芯片仅仅有少数引脚的标注字母不同,而对应的功能是
完全相同的。
用ADC0808,按照0809要求的时序进行仿真,完全可以仿真出ADC0809的效果。 一般的教材中,一提到ADC0809,往往都要讲:用D触发器对ALE分频来提供CLK。 实际上,为ADC0809提供CLK脉冲,完全可以利用单片机闲置的引脚,大可不必另外使用硬件。
有些人,也认识到了这个问题,他们采用了单片机定时器中断来输出脉冲,省掉了硬件D触发器。
其实,用定时器,还是很浪费的。完全使用软件,也可以达到输出周期性脉冲的目的。
做而论道注意到:
在单片机软件中,基本上都要编写延时函数。
在延时函数中实现输出CLK脉冲,就可以既不用外接硬件,也不占用单片机本身的硬件资源,同样也达到了延时目的,可谓一举多得。
这个做法,在目前所见到各种书籍、各个网文中,还没有人提到 ! 这个做法,可说是做而论道的独创。 呵呵,颇为自豪。
下面就是利用ADC0808仿真ADC0809的画面:
图片链接:
http://hi.baidu.com/%D7%F6%B6%F8%C2%DB%B5%C0/album/item/e0e4a8b40ee7d1a637d3ca3c.html#
图中右边的拨动开关用于选择模拟通道,选好的通道号码将在数码显示器的最高位显示。
图中左边,可以输入8路模拟信号,但是限于幅面,仅仅画出了一路,读者可自行加入另外的7路。 图中下方是一个频率计,测量出CLK的频率约有50KHz,在这个频率下,ADC0809大约用10ms即可进行一次转换,速度足够快了。
适当修改延时函数达到参数,输出CLK的频率还可以高出很多。 C语言程序如下:
#include //--------------------------------------------------- sbit ALE = P3^3; sbit OE = P3^4; sbit EOC = P3^5; sbit STA = P3^6; sbit CLK = P3^7;
//--------------------------------------------------- unsigned int num; //AD转换后的数字量 char ch; //通道号码
//--------------------------------------------------- void delay(unsigned int z) //延时函数 {
unsigned int x, y;
for(x = z; x > 0; x--) for(y = 110; y > 0; y--) CLK = ~CLK; } //在延时函数中,给ADC0809送去CLK,也算是做而论道的一个独创吧 ! //--------------------------------------------------- void display(void) {
char code table[] = { //共阴段码 0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x07, //0~7 0x7f, 0x6f, 0x77, 0x7c, 0x39, 0x5e, 0x79, 0x71}; //8~F
P0 = 0; P2 = 0xf7; P0 = table[ch]; delay(2);//通道号码
P0 = 0; P2 = 0xfb; P0 = table[num / 100] + 128; delay(2);//百位加上小数点
P0 = 0; P2 = 0xfd; P0 = table[num % 100 / 10]; delay(2);//十位 P0 = 0; P2 = 0xfe; P0 = table[num % 10]; delay(2);//个位 P0 = 0; //关闭显示器 }
//--------------------------------------------------- void main(void) {
while(1) { //无穷循环 ch = P2 / 32; //取来通道号码 P3 = ch + 0xf8; //送给ADC0809 ALE = 1; ALE = 0; //地址锁存
STA = 1; STA = 0; //开始转换,稍候,才可读EOC display(); //显示,既做延时,又输出CLK while(EOC == 0) display(); //等待转换结束
num = P1; //取出转换结果
num = num * 100 / 51; //比例变换:255 --> 500