自由空间

free space

  VC知识库BLOG :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 登录 ::
  82 随笔 :: 1 文章 :: 237 评论 :: 0 Trackbacks
<2010年3月>
28123456
78910111213
14151617181920
21222324252627
28293031123
45678910

留言簿(23)

随笔分类

随笔档案

文章档案

相册

vckbase

搜索

最新评论

阅读排行榜

评论排行榜

本文主要介绍并行端口的结构以及简单的对并口的读、写并以及如何获得端口的状态。   

并行接口的分类: SPP(标准并行接口) ,EPP(增强型并行接口),ECP(扩展型并行端口)

  标准并行端口(SPP)也是最早的端口定义,主要功能如下,1:并行端口提供了8个数据线以进行并行的字节传输,2:计算机能够通过数据线向打印机发送选能信号,以通知打印机已经准备好接收数据,3:打印机招收到数据后,向计算机发送一个回应信号(NACK)。其各位信号线所代表的意义详见下表。

  增强型并行端口(EPP)的出现提供了一种更高性能的连接方式,并东路向下兼容所有在此之前存在的并行接口及外设。与SPP不同之处在于原来17个信号中的重新定义,在这17个信号中,EPP使用了其中的14个信号进行传输,握手和选通,剩下的3个信号可以由外设设计者有来自定义。

  并行接口的大致结构:
     
并行口一般有25个引脚,其中包括8位数据线,5位打印机状态线,4位控制线.下面将对这些引脚予以详细说明:
  (注:1:>出,表示由计算机发向打印机;入,表示由打印机发向计算机,
      2:>低电平有效信号用上划线或星号表示(如S7*),高电平有效信号则没有上划线或星号)
      

引脚号

名称

数据位 寄存器

数据方向

定义

1

/STROBE

C0*  

选通信号,低电平有效信号,表明线上有数据到达.

2

D0

 DATA_1-DATA_8

 

 

 

 

 

 

D1-D8

 

 

  八位数据线,只有在SPP指令下才有能输出数据.

 

 

 

 

3

D1

4

D2

5

D3

6

D4

7

D5

8

D6

9

D7

 

10

/ACK

 S6 Status

应答,以插入低电平的形式出现,表明最后一个字符已招收完毕。

11

BUSY

 

S7*

Status

繁忙通知,以插入高电平的方式出现,表明打印机处于忙状态不能再接收数据。

12

PE

S5 Status

没有打印机纸。

13

SELECT

 S4 Status

选择输入,以插入高电平的方式出现,表明打印机处于在线待命状态。

14

AUTO FEED

 C1* Control

自动馈送,低电平有效信号民,通知打印机对于每遇到一个回车进行自动换行。

15

/ERROR

S3 Status

错误,该信号由打印机发向计算机,表明打印机处于错误状态。

16

/INIT

 C2 Control

初始化,低电平有效信号,该信号用来对打印机进行复位。

17

/SELIN

 C3* Control

选择输入,低电平有效信号,表明已经选中的打印机。

18

GND

   

---

信号接地

19

GND

   

---

信号接地

20

GND

   

---

信号接地

21

GND

   

---

信号接地

22

GND

   

---

信号接地

23

GND

   

---

信号接地

24

GND

   

---

信号接地

25

GND

   

---

信号接地

    打印机常用端口: 对于LPT1:0X378 为数据发送地址,0X379为打印机状态地址,0X37A为计算机向打印机控制地址, 通常为了使程序具有通用性我们可以从注册表中取得这个地址,对于WINDOWS CE 而言,这个值被存储在{HKEY_LOCAL_MACHINE//DRIVERS//BUILTIN//PARALLEL//IOBASE}.然后我们就可以通过对此三个端口进行控制达到简单的并口编程的目的.

下面是对打印机状态端口及打印机控制端口作一详细解释:

控制端口:

0x37A

1

1

输入控制

中断

17线

16 线

14 线

1  线

  这个0x37A计算机控制打印机的地址,可以产生对打印机进行控制的必要信号,可写,两高位(7和8)没什么用,第6位写1表示可以向并口输出数据。第五位中断信号(IRQ EN),利用此信号线,驱动程序可以在STATUS端口信号(nAck)的帮助下,使用该信号对中断信号的产生与否进行控制。第3210位分别控制第17线,第16线,第14线和第1线。(可以控制它们的状态)具体作用可参见前表

状态端口:

0x379

(S7)11

(S6)10应答

(S5)12缺纸

(S4)13联机

(S3)15 错误

  S2

S1 

S0 

  0x379为打印机状态地址,可读,通过个端口打印机适配器可以很方便的读取打印机的状态。

标记为S7的信号表示最高位,SO表示为最低位,只有S3-S7五个信号才是真正有用的信号。他们的具体信号功能解释如下:

S7*(busy):打印机使用该信号表示打印机正处于忙状态,不能再接收数据。需要强调的是,该信号通过适配器板时,进行了反相处理,因此连接器上的低电平到微处理器时就变成了高电平。

S6(nAck):当适配器发出选通信号时,打印机就会产生该信号作为响应。通常情况下,该信号是高电平,选通打印机之后,打印机首先把该信号设为低电平,然后再返回高电平。

S5(PE):当打印机缺纸时,它就会产生一个这样的信号,通常情况下,该信号由打印机保持为低电平,打印机纸使用完之后,该信号就会变成高电平。

S4(SELECT):当打印机恢复正常操作时,它就会插入一个高电平的该信号。当打印机处于无效状态时,访信号就会变成低电平。

S3(NERROR):当打印机出现错误时就会产生这种邮错信号。产生出错的原因很多,如打印纸堵了或产生了内部错误。产生错误时该信号就会设置成低电平。

以下为在Windows CE 下打印机各种状态时,所对应的状态寄存器的(AL)的值:

1:>在没接入打印机时寄存器AL值为127,对应二进制是:1111111
2:>打印机在缺纸灯不亮时寄存器AL值为144,对应二进制是:10010000
3:>打印机在缺纸灯亮时寄存器AL值为 119;对应二进制值是:11101111
4:>打印机在不缺纸的情况下寄存器AL值 223,对应二进制值是:11011111
5:>打印机在没开机的情况下得到AH值为207,对应二进制值是:11001111 

下面介绍对并口的编程控制:

(编程控制示例)(为汇编代码)
// 此段代码为并口向打印机进行写数据,并发送控制信息。

#define LPT_CLEAR_MASK 0x40
#define LPT_STROBE_HI 0x0D
#define LPT_STROBE_LO 0x0C

#define LPT_STATUS_BITS 0xF8
#define LPT_BITS_INVERT 0x48
#define LPT_NOTBUSY 0x80
#define LPT_PAPEROUT 0x20
#define LPT_SELECT 0x10
#define LPT_NFAULT 0x08

#define LPT_TimeOut 0x01

void OutByte(ULONG dataport, BYTE databyte) {
#if x86
_asm {
mov dx, word ptr [dataport]
mov al, byte ptr [databyte]
out dx, al
out dx, al
add dx, 2
in al, dx
and al, LPT_CLEAR_MASK
mov cl, al
or al, LPT_STROBE_HI
out dx, al
out dx, al
out dx, al
out dx, al
or cl, LPT_STROBE_LO
mov al, cl
out dx, al
}
#else


WRITE_PORT_UCHAR((PUCHAR)dataport,databyte);

dataport+=2;
databyte = (READ_PORT_UCHAR((PUCHAR)dataport) & LPT_CLEAR_MASK)
| LPT_STROBE_HI;
WRITE_PORT_UCHAR((PUCHAR)dataport,databyte);

databyte = (READ_PORT_UCHAR((PUCHAR)dataport) & LPT_CLEAR_MASK)
| LPT_STROBE_LO;
WRITE_PORT_UCHAR((PUCHAR)dataport,databyte);

#endif
}

// 此段代码为读取打印机当前状态。
BYTE CheckStatus(unsigned port) {
BYTE bRet;

#if x86
_asm {
mov dx, word ptr[port]
Checkme:
in al, dx
mov ah, al
nop
nop
in al, dx
cmp al, ah
jnz Checkme
and ah, LPT_STATUS_BITS
xor ah, LPT_BITS_INVERT
test ah, LPT_PAPEROUT or LPT_NFAULT
jnz CS_HasError
test ah, LPT_SELECT
jz CS_HasError
and ah, LPT_NOTBUSY
jz CS_HasError
xor eax, eax
CS_HasError:
mov [bRet],al
}
#else

BYTE bStatus;
do {
bRet= bStatus= READ_PORT_UCHAR((PUCHAR)port);
} while (bStatus != READ_PORT_UCHAR((PUCHAR)port));
bStatus&= LPT_STATUS_BITS;
bStatus^= LPT_BITS_INVERT;

if (!(bStatus & (LPT_PAPEROUT | LPT_NFAULT)) &&
(bStatus & LPT_SELECT) && (bStatus & LPT_NOTBUSY))
bRet=0;
#endif
return bRet;
}

// The End

posted on 2005-05-12 14:24 自由空间 阅读(12508) 评论(8)  编辑 收藏

评论

# re: 转:并口编程参考资料 2005-05-12 16:22 紫水晶
汇编???

# re: 转:并口编程参考资料 2005-05-13 11:24 freedk
用C也一样嘛,汇编再说查资料也能看得懂。知道原理就行了,什么语言实现到不重要

# re: 转:并口编程参考资料 2005-08-01 09:15 guaidan1
有没有vc++编写的并口编程?各位高手谁有的话给我e-mail一份!
小弟感激涕零!!lxjmike@hotmail.com

# re: 转:并口编程参考资料 2006-01-23 16:31 张修军
您好,我现在碰到一个比较棘手的问题。
就是用电脑监听打印数据。开始我用VC++6采用windriver的EPP中断方式来做,对某一种机器是可以的。但是拿到其他地方做,被监听的机器不是死机就是打印乱码(连接线我们确定没有问题)。您看这可能是什么原因造成的呢?是不是要加什么光电隔离?我的连线是直接一分二硬连的。
如果采用单片机方式将并口转换成串口,电脑直接读串口也是可行的办法,但我没有硬件的能力。不知道您哪有没有现有的程序?能给我一份的话,我万分感激。
我的EMAIL:woodszhang@gmail.com
电话:0851-7849462 张修军

# re: 转:并口编程参考资料 2006-08-03 13:08 海浪
我用VC能实现并口的单字节的通信 ,可当我想实现多字节传输是就出现问题,反馈信号无法获取,我用的是EPP1.7,请高手指点。Email:taozifei@tom.com

# re: 转:并口编程参考资料 2006-09-23 06:01 darkfox
你这个方法恐怕现在已经不适用了,xp下肯定不行
没有ring0权限 无法使用 in out 指令(虚拟86好象可以)

我这里有个方法,就是使用API CreateFile 函数,找遍了资料,只提
到可以用来操作串口,但实际上还可以操作 LPT 口,下面的例子原来
是属于串口通讯的,被我改了下,很简单,只需要在函数的
第一个参数
里填入 "LPT1" (如果你还有第2个口的话,也可以是"LPT2")
测试后,发现的确能打开并口,测试方法很简单,把下面的代码放
到任何一个过程中,轮番给这个函数的第一个参数传入 :
"LPT2" 会提示出错(没有这个口的情况下)
"LPT1" 正常打开,没有提示

下面这段例子来源于VC,但我用的是BCB,改过后稍有不同,VC里的话再改下就行了。

/*    并口操作
       HANDLE hCom;
        DWORD dwError;
        AnsiString portname = Edit1->Text; //Edit1->Text = "LPT1"
        hCom=CreateFile(portname.c_str(), // 文件名
            GENERIC_READ | GENERIC_WRITE, // 允许读和写
            0, // 独占方式
            NULL,
            OPEN_EXISTING, //打开而不是创建
            FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, // 重叠方式
            NULL
            );
        if(hCom == INVALID_HANDLE_VALUE)
        {   dwError=GetLastError();
            ShowMessage("有错误");
        }
        if(hCom)
            CloseHandle(hCom);

*/

# re: 转:并口编程参考资料 2008-12-05 10:21 hyz
楼主的方法我在XP系统下运行,报找不到WRITE_PORT_UCHAR,和READ_PORT_UCHAR函数的错误,是不是系统不支持用这种方法读取状态端口呢,要怎么才可以读取状态端口的数据呢。
用CreateFile打开端口后,可以使用WriteFile函数向打印机发送命令,但无法用ReadFile函数从打印机读取数据。

# re: 转:并口编程参考资料 2008-12-20 09:10 飞鸽传书
呵呵,代码很不错。_ASM

标题  
姓名  
主页
验证码 *
内容   
  登录  使用高级评论  Top
[使用Ctrl+Enter键可以直接提交]