ADC0808/ADC0809是帶有8位A/D轉換器、8路多路開關以及微處理機兼容的控制邏輯的CMOS組件。它是逐次逼近式A/D轉換器,可以和單片機直接接口。 (1). ADC0809的內部邏輯結構file:///C:/Users/DELL/AppData/Local/Temp/msohtmlclip1/01/clip_image001.gif 由上圖可知,ADC0809由一個8路模擬開關、一個地址鎖存與譯碼器、一個A/D轉換器和一個三態輸出鎖存器組成。多路開關可選通8個模擬通道,允許8路模擬量分時輸入,共用A/D轉換器進行轉換。三態輸出鎖器用于鎖存A/D轉換完的數字量,當OE端為高電平時,才可以從三態輸出鎖存器取走轉換完的數據。 (2). 引腳結構 file:///C:/Users/DELL/AppData/Local/Temp/msohtmlclip1/01/clip_image002.jpg IN0-IN7:8條模擬量輸入通道 ADC0809對輸入模擬量要求:信號單極性,電壓范圍是0-5V,若信號太小,必須進行放大;輸入的模擬量在轉換過程中應該保持不變,如若模擬量變化太快,則需在輸入前增加采樣保持電路。 地址輸入和控制線:4條 ALE為地址鎖存允許輸入線,高電平有效。當ALE線為高電平時,地址鎖存與譯碼器將A,B,C三條地址線的地址信號進行鎖存,經譯碼后被選中的通道的模擬量進轉換器進行轉換。A,B和C為地址輸入線,用于選通IN0-IN7上的一路模擬量輸入。通道選擇表如下表所示。 C | B | A | 選擇的通道 | 0 | 0 | 0 | IN0 | 0 | 0 | 1 | IN1 | 0 | 1 | 0 | IN2 | 0 | 1 | 1 | IN3 | 1 | 0 | 0 | IN4 | 1 | 0 | 1 | IN5 | 1 | 1 | 0 | IN6 | 1 | 1 | 1 | IN7 |
數字量輸出及控制線:11條 ST為轉換啟動信號。當ST上跳沿時,所有內部寄存器清零;下跳沿時,開始進行A/D轉換;在轉換期間,ST應保持低電平。EOC為轉換結束信號。當EOC為高電平時,表明轉換結束;否則,表明正在進行A/D轉換。OE為輸出允許信號,用于控制三條輸出鎖存器向單片機輸出轉換得到的數據。OE=1,輸出轉換得到的數據;OE=0,輸出數據線呈高阻狀態。D7-D0為數字量輸出線。 CLK為時鐘輸入信號線。因ADC0809的內部沒有時鐘電路,所需時鐘信號必須由外界提供,通常使用頻率為500KHZ, VREF(+),VREF(-)為參考電壓輸入。 2.ADC0809應用說明 (1). ADC0809內部帶有輸出鎖存器,可以與AT89S51單片機直接相連。 (2). 初始化時,使ST和OE信號全為低電平。 (3). 送要轉換的哪一通道的地址到A,B,C端口上。 (4). 在ST端給出一個至少有100ns寬的正脈沖信號。 (5). 是否轉換完畢,我們根據EOC信號來判斷。 (6). 當EOC變為高電平時,這時給OE為高電平,轉換的數據就輸出給單片機了。 3.實驗任務 如下圖所示,從ADC0809的通道IN3輸入0-5V之間的模擬量,通過ADC0809轉換成數字量在數碼管上以十進制形成顯示出來。ADC0809的VREF接+5V電壓。 4.電路原理圖 file:///C:/Users/DELL/AppData/Local/Temp/msohtmlclip1/01/clip_image004.jpg P3.7上的開關無用。不要SUN7474來分頻,直接在ADC0808的CLK接500KHz。在數碼管接上拉電阻。 圖1.27.1 5. 系統板上硬件連線 (1). 把“單片機系統板”區域中的P1端口的P1.0-P1.7用8芯排線連接到“動態數碼顯示”區域中的A B C D E F G H端口上,作為數碼管的筆段驅動。 (2). 把“單片機系統板”區域中的P2端口的P2.0-P2.7用8芯排線連接到“動態數碼顯示”區域中的S1 S2 S3 S4 S5 S6 S7 S8端口上,作為數碼管的位段選擇。 (3). 把“單片機系統板”區域中的P0端口的P0.0-P0.7用8芯排線連接到“模數轉換模塊”區域中的D0D1D2D3D4D5D6D7端口上,A/D轉換完畢的數據輸入到單片機的P0端口 (4). 把“模數轉換模塊”區域中的VREF端子用導線連接到“電源模塊”區域中的VCC端子上; (5). 把“模數轉換模塊”區域中的A2A1A0端子用導線連接到“單片機系統”區域中的P3.4 P3.5 P3.6端子上; (6). 把“模數轉換模塊”區域中的ST端子用導線連接到“單片機系統”區域中的P3.0端子上; (7). 把“模數轉換模塊”區域中的OE端子用導線連接到“單片機系統”區域中的P3.1端子上; (8). 把“模數轉換模塊”區域中的EOC端子用導線連接到“單片機系統”區域中的P3.2端子上; (9). 把“模數轉換模塊”區域中的CLK端子用導線連接到“分頻模塊”區域中的 /4 端子上; (10). 把“分頻模塊”區域中的CK IN端子用導線連接到“單片機系統”區域中的 ALE 端子上; (11). 把“模數轉換模塊”區域中的IN3端子用導線連接到“三路可調壓模塊”區域中的 VR1 端子上; 6. 程序設計內容 (1). 進行A/D轉換時,采用查詢EOC的標志信號來檢測A/D轉換是否完畢,若完畢則把數據通過P0端口讀入,經過數據處理之后在數碼管上顯示。 (2). 進行A/D轉換之前,要啟動轉換的方法: ABC=110選擇第三通道 ST=0,ST=1,ST=0產生啟動轉換的正脈沖信號 7. 匯編源程序 CH EQU30H DPCNT EQU31H DPBUF EQU33H GDATA EQU32H ST BITP3.0 OE BITP3.1 EOC BITP3.2
ORG00H LJMPSTART ORG0BH LJMPT0X ORG30H START: MOVCH,#0BCH MOV P3,CH //通道3 MOVDPCNT,#00H MOVR1,#DPCNT MOVR7,#5 MOVA,#10 MOVR0,#DPBUF LOP: MOV@R0,A INCR0 DJNZR7,LOP MOV@R0,#00H INCR0 MOV@R0,#00H INCR0 MOV@R0,#00H MOVTMOD,#01H MOVTH0,#(65536-4000)/256 MOVTL0,#(65536-4000) MOD 256 SETBTR0 SETBET0 SETBEA WT: CLRST SETBST CLRST WAIT: JNBEOC,WAIT SETBOE MOVGDATA,P0 CLROE movdptr,#tab ;將AD轉換結果轉換成BCD碼 mov a,GDATA movc a,@a+dptr movb,#2 div ab movr1,b movb,#10 div ab mov 38H,a mov 39H,b cjne r1,#01,kk1 mov 3AH,#05 back: SJMP WT kk1: mov 3AH,#00 ajmp back //MOVA,GDATA //MOVB,#100 //DIVAB //MOV33H,A //MOVA,B //MOVB,#10 //DIV AB //MOV34H,A //MOV35H,B //SJMPWT T0X: NOP MOVTH0,#(65536-350)/256 MOVTL0,#(65536-350) MOD 256 MOV A,#0FFH MOV P2,A //清屏 MOV A,DPCNT CJNE A,#5,NEX MOV DPTR,#DPCD MOV A,DPCNT ADD A,#DPBUF MOV R0,A MOV A,@R0 MOVC A,@A+DPTR ORL A,#080H MOV P1,A SJMP NEX1 NEX: MOV DPTR,#DPCD MOVA,DPCNT ADDA,#DPBUF MOVR0,A MOVA,@R0 MOVCA,@A+DPTR MOVP1,A NEX1: MOV DPTR,#DPBT MOVA,DPCNT MOVCA,@A+DPTR MOVP2,A INCDPCNT MOVA,DPCNT CJNEA,#8,NEXT MOVDPCNT,#00H NEXT: RETI DPCD: DB3FH,06H,5BH,4FH,66H DB6DH,7DH,07H,7FH,6FH,00H DPBT: DB0FEH,0FDH,0FBH,0F7H DB0EFH,0DFH,0BFH,07FH . tab: db 0, 0, 0, 1, 0, 2, 0, 0, 3, 0 db 4, 0, 0, 5, 0, 6, 0, 0, 7, 0 db 8, 0, 0, 9, 0, 0, 10, 0, 11, 0 db 0, 12, 0, 13, 0, 0, 14, 0, 15, 0 db 0, 16, 0, 17, 0, 0, 18, 0, 19, 0 db 0, 20, 0, 0, 21, 0, 22, 0, 0, 23 db 0, 24, 0, 0, 25, 0, 26, 0, 0, 27 db 0, 28, 0, 0, 29, 0, 0, 30, 0, 31 db 0, 0, 32, 0, 33, 0, 0, 34, 0, 35 db 0, 0, 36, 0, 37, 0, 0, 38, 0, 39 db 0, 0, 40, 0, 0, 41, 0, 42, 0, 0 db 43, 0, 44, 0, 0, 45, 0, 46, 0, 0 db 47, 0, 48, 0, 0, 49, 0, 50, 0, 0 db 51, 0, 0, 52, 0, 53, 0, 0, 54, 0 db 55, 0, 0, 56, 0, 57, 0, 0, 58, 0 db 59, 0, 0, 60, 0, 0, 61, 0, 62, 0 db 0, 63, 0, 64, 0, 0, 65, 0, 66, 0 db 0, 67, 0, 68, 0, 0, 69, 0, 70, 0 db 0, 71, 0, 0, 72, 0, 73, 0, 0, 74 db 0, 75, 0, 0, 76, 0, 77, 0, 0, 78 db 0, 79, 0, 0, 80, 0, 0, 81, 0, 82 db 0, 0, 83, 0, 84, 0, 0, 85, 0, 86 db 0, 0, 87, 0, 88, 0, 0, 89, 0, 90 db 0, 0, 91, 0, 0, 92, 0, 93, 0, 0 db 94, 0, 95, 0, 0, 96, 0, 97, 0, 0 db 98, 0, 99, 0, 0, 100, 0, 0, 0 END
8. C語言源程序1 #include <AT89X52.H> unsigned char code dispbitcode[]={0xfe,0xfd,0xfb,0xf7, 0xef,0xdf,0xbf,0x7f}; unsigned char code dispcode[]={0x3f,0x06,0x5b,0x4f,0x66, 0x6d,0x7d,0x07,0x7f,0x6f,0x00}; unsigned chardispbuf[8]={10,10,10,10,10,0,0,0}; unsigned char code tab[]={0,0, 0, 1, 0, 2, 0, 0, 3, 0, 4, 0, 0, 5, 0, 6, 0, 0, 7, 0 , 8, 0, 0, 9, 0, 0, 10, 0, 11, 0 , 0, 12, 0, 13, 0, 0, 14, 0, 15, 0 , 0, 16, 0, 17, 0, 0, 18, 0, 19, 0, 0, 20, 0, 0, 21, 0, 22, 0, 0, 23, 0, 24, 0, 0, 25, 0, 26, 0, 0, 27, 0, 28, 0, 0, 29, 0, 0, 30, 0, 31, 0, 0, 32, 0, 33, 0, 0, 34, 0, 35, 0, 0, 36, 0, 37, 0, 0, 38, 0, 39, 0, 0, 40, 0, 0, 41, 0, 42, 0, 0, 43,0, 44, 0, 0, 45, 0, 46, 0, 0, 47,0, 48, 0, 0, 49, 0, 50, 0, 0, 51,0, 0, 52, 0, 53, 0, 0, 54, 0 , 55,0, 0, 56, 0, 57, 0, 0, 58, 0, 59,0, 0, 60, 0, 0, 61, 0, 62, 0, 0, 63, 0, 64, 0, 0, 65, 0, 66, 0, 0, 67, 0, 68, 0, 0, 69, 0, 70, 0, 0, 71, 0, 0, 72, 0, 73, 0, 0, 74, 0, 75, 0, 0, 76, 0, 77, 0, 0, 78, 0, 79, 0, 0, 80, 0, 0, 81, 0, 82, 0, 0, 83, 0, 84, 0, 0, 85, 0, 86, 0, 0, 87, 0, 88, 0, 0, 89, 0, 90, 0, 0, 91, 0, 0, 92, 0, 93, 0, 0, 94,0, 95, 0, 0, 96, 0, 97, 0, 0, 98,0, 99, 0, 0, 100,0, 0, 0}; unsigned char dispcount; unsigned int R1; sbit ST=P3^0; sbit OE=P3^1; sbit EOC=P3^2; unsigned char channel=0xbc; //1011 1100,ABC=110選擇第三通道同時使ST=0,OE=0輸出數據線呈高阻狀態,EOC=1。 unsigned char getdata; void main(void) { TMOD=0x01; TH0=(65536-4000)/256; TL0=(65536-4000)%256; TR0=1; ET0=1; EA=1; P3=channel;
while(1) { ST=0; ST=1; ST=0; while(EOC==0); OE=1; getdata=P0; OE=0; getdata=tab[getdata]; R1=getdata%2; getdata=getdata/2; dispbuf[5]=getdata/10; dispbuf[6]=(getdata%10); if(R1==1) dispbuf[7]=5; else dispbuf[7]=0; //dispbuf[2]=getdata/100; //getdata=getdata%10; //dispbuf[1]=getdata/10; //dispbuf[0]=getdata%10; } } void t0(void) interrupt 1 using 0 { TH0=(65536-350)/256; TL0=(65536-350)%256; P2=0xff; if(dispcount==5) {P1=dispcode[dispbuf[dispcount]]|0x80; P2=dispbitcode[dispcount]; } else {P1=dispcode[dispbuf[dispcount]]; P2=dispbitcode[dispcount]; } //P1=dispcode[dispbuf[dispcount]]; //P2=dispbitcode[dispcount]; dispcount++; if(dispcount==8) { dispcount=0; } } 9. C語言源程序 2 #include <AT89X52.H> unsigned char codedispbitcode[]={0xfe,0xfd,0xfb,0xf7, 0xef,0xdf,0xbf,0x7f}; unsigned char code dispcode[]={0x3f,0x06,0x5b,0x4f,0x66, 0x6d,0x7d,0x07,0x7f,0x6f,0x00}; unsigned chardispbuf[8]={10,10,10,10,10,10,10,10};
unsigned char dispcount;
sbit ST=P3^0; sbit OE=P3^1; sbit EOC=P3^2; unsigned char channel=0xbc; unsigned char getdata; long int i; //因為8位二進制最高為255,后面的i=getdata*196;最大值超過255,所以要定義更大的空間 (基于8051+ADC0809設計數字電壓表的匯編實現.doc) void main(void) { TMOD=0x01; TH0=(65536-4000)/256; TL0=(65536-4000)%256; TR0=1; ET0=1; EA=1; P3=channel;
while(1) { ST=0; ST=1; ST=0; while(EOC==0); OE=1; getdata=P0; OE=0; i=getdata*196; dispbuf[5]=i/10000; i=i%10000; dispbuf[6]=i/1000; i=i%1000; dispbuf[7]=i/100; } } void t0(void) interrupt 1 using 0 { TH0=(65536-6000)/256; TL0=(65536-6000)%256; P2=0xff; P1=dispcode[dispbuf[dispcount]]; P2=dispbitcode[dispcount]; dispcount++; if(dispcount==8) { dispcount=0; } }
|