|
本文所述的高分辨率A/D转换器摘自EDN杂志,它可以直接由RS232串口驱动。</P>
Edited by Bill Travis
Yongping Xia, Teldata, Los Angeles, CA -- EDN, 9/27/01</P>
通常情况下,PC机是通过插入ADC转换卡实现模拟信号处理的,而本文所述的A/D转换器则是通过RS232口直接驱动的。实际上,RS232口可以驱动18位的ADC转换器,串口不仅能提供控制信号,还能提供芯片所需电源。IC1是一款18位的MAX132 ADC转换器接口,DIN是输入信号,SCLK是时钟信号,DOUT是数据输出,EOC是转换完成标志。RS-232口有三条输出线Pin 3 (TX), Pin 4 (DTR), 和Pin 7 (RTS)。TX可用于产生SCLK信号并提供负电源,DTR用于发送数据,RTS提供CS片选信号及正电源。不论是正电源还是负电源,均使用比较大的电容存储能量。当TX产生一个脉冲信号,或DTR发送片选信号(低有效),电容向MAX132供电。MAX132几乎集成了全部需要的功能,但1.2V的参考电压必须由外部的LM385提供。MAX132的信号输入范围是-512 to +512 mV。</P>
</P>
< ><img src="attachments/dvbbs/2004-11/20041126102052806.gif" border="0" onload="if(this.width>screen.width*0.7) {this.resized=true; this.width=screen.width*0.7; this.alt=\'Click here to open new window\nCTRL+Mouse wheel to zoom in/out\';}" onmouseover="if(this.width>screen.width*0.7) {this.resized=true; this.width=screen.width*0.7; this.style.cursor=\'hand\'; this.alt=\'Click here to open new window\nCTRL+Mouse wheel to zoom in/out\';}" onclick="if(!this.resized) {return true;} else {window.open(\'attachments/dvbbs/2004-11/20041126102052806.gif\');}" onmousewheel="return imgzoom(this);" alt="" /></P>
< >*******************************************************************************
; 列表1 ----- A/D转换及显示源程序
; LISTING 1 - SCREEN-DISPLAY ROUTINE FOR ANALOG-TO-DIGITAL-CONVERSION RESULTS
;
; "18-bit ADC uses PC's serial port," EDN, Sept 27, 2001, pg 84
;
;*******************************************************************************</P>
< >#include <stdio.h>
#include <dos.h>
#include <time.h>
#include <conio.h>
#include <bios.h>
#include <conio.h></P>
<P>#define COM1 0
#define MCR 4 /* control register */
#define MSR 6 /* status register */</P>
<P>int i, j, base_add1=0x3f8, base_add2=0x2f8, out_data=0x03, in_data[4];</P>
<P>float data;</P>
<P>void send_clk(void)
{
delay(1);
outportb(base_add1, 0x00);
delay(3);
}</P>
<P>void read_port(void)
{
int control[4], out_control;
data=0;
for (i=0; i<4; i++)
in_data<i>=0;
control[0]=0x82;
control[1]=0x04;
control[2]=0x00;
control[3]=0x00;
out_data|=0x02;
outportb(base_add1+MCR, out_data); /* CS high */
delay(10);
out_data&=0x01;
outportb(base_add1+MCR, out_data); /* CS low */
delay(10);
out_control=control[0];
for (i=0; i<8; i++)
{
if (out_control>=0x80)
out_data|=0x01;
else
out_data&=0x02;
outportb(base_add1+MCR, out_data);
send_clk(); /* clock out */
out_control&=0x7f;
out_control=out_control*2;
}
out_data|=0x02;
outportb(base_add1+MCR, out_data); /* CS high */
delay(10);
do{
}while((inportb(base_add1+MSR)&0x40)==0); /* waiting for EOC=high */
for (j=1; j<4; j++)
{
out_control=control[j];
in_data[j]=0;
out_data&=0x01;
outportb(base_add1+MCR, out_data); /* CS low */
delay(10);
for (i=0; i<8; i++)
{
if (out_control>=0x80)
out_data|=0x01;
else
out_data&=0x02;
outportb(base_add1+MCR, out_data);
in_data[j]=in_data[j]*2+(inportb(base_add1+MSR)&0x80)/0x80;
send_clk(); /* clock out */
out_control&=0x7f;
out_control=out_control*2;
}
out_data|=0x02;
outportb(base_add1+MCR, out_data); /* CE high */
delay(10);
}
if ((in_data[1]&0x08)==0)
data=(float)(in_data[1]&0x07)+(float)(in_data[2])*2048+(float)(in_data[3])*8;
else /* reading is negative */
{
in_data[1]=in_data[1]&0x07;
in_data[1]=(8-in_data[1])&0x07;
in_data[2]=(256-in_data[2])&0xff;
in_data[3]=(256-in_data[3])&0xff;
data=-((float)(in_data[1])+(float)(in_data[2])*2048+(float)(in_data[3])*8);
}
}</P>
<P>void dis_data(void)
{
float show_data;
show_data=0.000002*data;
gotoxy(1,1);
printf("%.5f ", show_data);
gotoxy(1,1);
}</P>
<P>void init(void)
{
bioscom(0, 255, COM1); /* set up COM1 */
out_data=0x02;
outportb(base_add1+MCR, out_data); /* CS=high, DIN=low */
delay(100);
}</P>
<P>void main(void)
{
clrscr();
init();
gotoxy(60,24);
printf("Hit any key to quit");
do{
read_port();
dis_data();
delay(500);
} while(!kbhit());
}
</P> |
本帖子中包含更多资源
您需要 登录 才可以下载或查看,没有帐号?注册
x
|