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驱动程序中最长的一个,莱昂对其有详细的讲解,再次不再赘述。

博客地址:http://blog.csdn.net/cszhao1980

博客专栏地址:http://blog.csdn.net/column/details/lions-unix.html

(莱昂氏unix源代码分析导读-50)LP11行式打印机的更多相关文章

  1. (莱昂氏unix源代码分析导读-49) 字符缓冲区

    by cszhao1980 同块设备一样,对字符设备的输入输出也是通过缓冲区来进行的.使用缓冲区有个额外 的好处,即以缓冲区为界,函数可分为高低两个层次.低层函数负责与实际设备交互, 而高层函数只与缓 ...

  2. Hadoop源代码分析

    http://wenku.baidu.com/link?url=R-QoZXhc918qoO0BX6eXI9_uPU75whF62vFFUBIR-7c5XAYUVxDRX5Rs6QZR9hrBnUdM ...

  3. Hadoop源代码分析(完整版)

    Hadoop源代码分析(一) 关键字: 分布式云计算 Google的核心竞争技术是它的计算平台.Google的大牛们用了下面5篇文章,介绍了它们的计算设施. GoogleCluster:http:// ...

  4. hostapd源代码分析(二):hostapd的工作机制

    [转]hostapd源代码分析(二):hostapd的工作机制 原文链接:http://blog.csdn.net/qq_21949217/article/details/46004433 在我的上一 ...

  5. Android系统进程Zygote启动过程的源代码分析

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6768304 在Android系统中,所有的应用 ...

  6. Android系统匿名共享内存Ashmem(Anonymous Shared Memory)驱动程序源代码分析

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6664554 在上一文章Android系统匿名共 ...

  7. Android应用Activity、Dialog、PopWindow、Toast窗体加入机制及源代码分析

    [工匠若水 http://blog.csdn.net/yanbober 转载烦请注明出处.尊重劳动成果] 1 背景 之所以写这一篇博客的原因是由于之前有写过一篇<Android应用setCont ...

  8. UiAutomator喷射事件的源代码分析

    上一篇文章<UiAutomator源代码分析之UiAutomatorBridge框架>中我们把UiAutomatorBridge以及它相关的类进行的描写叙述,往下我们会尝试依据两个实例将这 ...

  9. 新秀nginx源代码分析数据结构篇(四)红黑树ngx_rbtree_t

    新秀nginx源代码分析数据结构篇(四)红黑树ngx_rbtree_t Author:Echo Chen(陈斌) Email:chenb19870707@gmail.com Blog:Blog.csd ...

随机推荐

  1. Android UI-开源框架ImageLoader的完美例子

    Android开源框架ImageLoader的完美例子 2013年8月19日开源框架之Universal_Image_Loader学习 很多人都在讨论如何让图片能在异步加载更加流畅,可以显示大量图片, ...

  2. 强强合体:Docker版Kali Linux发布

    Kali Linux是一款开源的基于Debian的渗透测试专用操作系统,系统中包含一系列用于渗透测试的神器.最近,Kali的开发者们为喜爱Docker的童鞋们发布了新版本. FreeBuf百科:什么是 ...

  3. [PHP] - 逗号和点号的区别

    比如:1. echo 'abc'.'def'; //用点号连接字符串 2. echo 'abc','def'; //用逗号连接字符串 也许很多人都知道逗号要比点号快.但是不知道为什么.更不知道这两者到 ...

  4. Monitor vs WaitHandle

    http://stackoverflow.com/questions/1355398/monitor-vs-waithandle-based-thread-sync A problem with Mo ...

  5. poj 1789 Truck History(最小生成树)

    模板题 题目:http://poj.org/problem?id=1789 题意:有n个型号,每个型号有7个字母代表其型号,每个型号之间的差异是他们字符串中对应字母不同的个数d[ta,tb]代表a,b ...

  6. C# 分布式缓存服务器方案

  7. NOI2013矩阵游戏

    Description 婷婷是个喜欢矩阵的小朋友,有一天她想用电脑生成一个巨大的n行m列的矩阵(你不用担心她如何存储).她生成的这个矩阵满足一个神奇的性质:若用F[i][j]来表示矩阵中第i行第j列的 ...

  8. quartz Cron表达式一分钟教程

    CronTrigger CronTriggers往往比SimpleTrigger更有用,如果您需要基于日历的概念,而非SimpleTrigger完全指定的时间间隔,复发的发射工作的时间表.CronTr ...

  9. 深入理解Arrays.sort()

    两种方法: 1.类本来就实现java.lang.Comparable接口,使类本身就有比较能力.接口实现compareTo方法,次方法接收另一个Object为参数,如果当前对象小于参数则返回负值,如果 ...

  10. 【JS】打印Excel——ActiveX控件

    function viewToExcel(){ var filepath = "f:\\PrinterExcel.xls"; var xlApp; var xlBook; var ...