by cszhao1980
LP11有两个设备寄存器:状态寄存器(lpsr)和数据缓冲寄存器(lpbuf),可通过以下结构进行访问:
8812: #define LPADDR 0177514
8823: struct {
8824: int lpsr;
8825: int lpbuf;
8826: };
(1). LPADDR.lpsr:状态寄存器
i. 第15位(最高位):出错标记;
如电源未开、缺纸、温度太高等都会造成标志置位。
判断条件: LPADDR.lpsr < 0
ii. 第7位: DONE标记(当该打印机控制器准备接收下一字符时设置);
iii. 第6位: Enable标记,设置后才启动了该打印机,其作用是使“DONE”或“ERROR”造成一中断。
(2). LPADDR.lpbuf:数据缓冲寄存器
第6至第0位保存要打印字符的7位ASCII代码,对此寄存器只能执行写操作。
除此之外,unix v6使用struct lp11来维护LP11设备信息,如下所示:
8829: struct {
8830: int cc; //前三个字用作缓冲头,即起到struct clist的作用。
8831: int cf;
8832: int cl;
8833: int flag; //记录LP11的工作状态信息
8834: int mcc;
8835: int ccc;
8836: int mlc;
8837: } lp11;
1 LP11的工作方式
【注】:莱昂书中并没有详细介绍LP11的工作方式——它实在是一种简单的设备,而且,在当时
也可能很常用。本章的部分内容是通过代码所进行的合理的推测。
首先, LP11设备拥有一个缓存区域,用来存放要打印的内容。向LP11发送数据是通过设置LPADDR.
lpbuf进行的,当我们设置LPADDR.lpbuf后,其内容会被LP11以很快的速度(超过CPU存放速度)存放
在其内部的缓存区域内。当LP11无法再接收数据时,会将DONE标记复位,此时,不应再向LP11发送数据。
当LP11收到足够多的数据后(也许是达到一整行的内容后),会启动打印操作。操作结束后,会通过中断
矢量200进入中断处理程序lpint。
2 LP11驱动程序
了解了LP11的工作方式之后,其驱动程序就很容易理解了。
其open和close代码都非常简单,在这里不再多解释了。唯一需要注意的是open时对“OPEN”flag的检查和设置,
这些代码实现了unix v6对LP11设备的互斥访问。
下面,我们来看一下其打印(write)过程。前面说过,write会通过writei函数调用到设备的实际write函数,对
LP11来说,是lpwrite。
8870: lpwrite()
8871: {
8872: register int c;
8873:
8874: while ((c=cpass())>=0) //从用户空间内取要打印的数据
8875: lpcanon(c); //简单起见,我们暂时将其看作一个lpoutput调用
8876: }
8986: lpoutput(c)
8987: {
8988: if (lp11.cc >= LPHWAT) //lp11起到clist的作用,cc为字符缓冲区内的char的数量。
8989: sleep(&lp11, LPPRI); //最多存放LPHWAT个,否则睡眠
8990: putc(c, &lp11); //向字符缓冲区放置char
8991: spl4();
8992: lpstart(); //启动真正的LP11驱动程序lpstart
8993: spl0();
8994: }
lpstart非常简单,即在DONE标记置位时,从字符缓冲区取字符发送给LP11设备。
8967: lpstart()
8968: {
8969: register int c;
8970:
8971: while (LPADDR->lpsr&DONE && (c = getc(&lp11)) >= 0)
8972: LPADDR->lpbuf = c;
8973: }
中断处理程序lpint也非常简单,如下所示:
8976: lpint()
8977: {
8978: register int c;
8979:
8980: lpstart();
8981: if (lp11.cc == LPLWAT || lp11.cc == 0)
8982: wakeup(&lp11);
8983: }
lpcanon是LP11驱动程序中最长的一个,莱昂对其有详细的讲解,再次不再赘述。