(莱昂氏unix源代码分析导读-50)LP11行式打印机
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行式打印机的更多相关文章
- (莱昂氏unix源代码分析导读-49) 字符缓冲区
by cszhao1980 同块设备一样,对字符设备的输入输出也是通过缓冲区来进行的.使用缓冲区有个额外 的好处,即以缓冲区为界,函数可分为高低两个层次.低层函数负责与实际设备交互, 而高层函数只与缓 ...
- Hadoop源代码分析
http://wenku.baidu.com/link?url=R-QoZXhc918qoO0BX6eXI9_uPU75whF62vFFUBIR-7c5XAYUVxDRX5Rs6QZR9hrBnUdM ...
- Hadoop源代码分析(完整版)
Hadoop源代码分析(一) 关键字: 分布式云计算 Google的核心竞争技术是它的计算平台.Google的大牛们用了下面5篇文章,介绍了它们的计算设施. GoogleCluster:http:// ...
- hostapd源代码分析(二):hostapd的工作机制
[转]hostapd源代码分析(二):hostapd的工作机制 原文链接:http://blog.csdn.net/qq_21949217/article/details/46004433 在我的上一 ...
- Android系统进程Zygote启动过程的源代码分析
文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6768304 在Android系统中,所有的应用 ...
- Android系统匿名共享内存Ashmem(Anonymous Shared Memory)驱动程序源代码分析
文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6664554 在上一文章Android系统匿名共 ...
- Android应用Activity、Dialog、PopWindow、Toast窗体加入机制及源代码分析
[工匠若水 http://blog.csdn.net/yanbober 转载烦请注明出处.尊重劳动成果] 1 背景 之所以写这一篇博客的原因是由于之前有写过一篇<Android应用setCont ...
- UiAutomator喷射事件的源代码分析
上一篇文章<UiAutomator源代码分析之UiAutomatorBridge框架>中我们把UiAutomatorBridge以及它相关的类进行的描写叙述,往下我们会尝试依据两个实例将这 ...
- 新秀nginx源代码分析数据结构篇(四)红黑树ngx_rbtree_t
新秀nginx源代码分析数据结构篇(四)红黑树ngx_rbtree_t Author:Echo Chen(陈斌) Email:chenb19870707@gmail.com Blog:Blog.csd ...
随机推荐
- 首次接触Winform前端交互
首次接触到在winform中加入网页,且跟前端脚本交互.找了一下这方面的资料 此博文转载原地址为:http://www.cnblogs.com/Charles2008/archive/2009/08/ ...
- window下编译ffmpeg 比较简单
网上关于编译ffmpeg的帖子很多,我也尝试了很多次,但是很多都过不了,一部分原因是版本问题,还有就是有的路劲没说的太明白导致的,经过一天的摸索,最终编译好了,下面把编译方式写下来,希望对看到帖子的人 ...
- SPRING IN ACTION 第4版笔记-第八章Advanced Spring MVC-007-给flowl加权限控制<secured>
States, transitions, and entire flows can be secured in Spring Web Flow by using the <secured> ...
- Spring中的Resource
Spring中的资源定义:Resource此接口的全名为:org.springframework.core.io.Resource比较常用的资源定义的实现类为:1.ClassPathResource ...
- ArrayList源代码深入剖析
第1部分 ArrayList介绍ArrayList底层采用数组实现,它的容量能动态增长.它继承于AbstractList,实现了List, RandomAccess, Cloneable, java. ...
- R语言学习笔记:查看函数的R源代码
getAnywhere 该函数可以返回一些函数的R源代码,如: getAnywhere(kmeans) 该函数具体用法,请参看官方说明. Retrieve an R Object, Including ...
- php.ini – 配置文件详解
详见: https://my.oschina.net/miaowang/blog/299546 这个文件必须命名为''php.ini''并放置在httpd.conf中的PHPIniDir指令指定的目录 ...
- poj2886Who Gets the Most Candies? (约瑟夫环)
http://poj.org/problem?id=2886 单点更新 初始位置都是1 如果这个人出去 位置变为0 利用线段树求区间k值 k值的计算如下 如果这个数值是负的 那么下一个人的就是((k- ...
- UVa 10214 (莫比乌斯反演 or 欧拉函数) Trees in a Wood.
题意: 这道题和POJ 3090很相似,求|x|≤a,|y|≤b 中站在原点可见的整点的个数K,所有的整点个数为N(除去原点),求K/N 分析: 坐标轴上有四个可见的点,因为每个象限可见的点数都是一样 ...
- BZOJ2893: 征服王
题解: 裸的上下界最小流是有问题的.因为在添加了附加源之后求出来的流,因为s,t以及其它点地位都是平等的.如果有一个流经过了s和t,那么总可以认为这个流是从s出发到t的满足题意的流. 既然可能存在s到 ...