(莱昂氏unix源代码分析导读-49) 字符缓冲区
by cszhao1980
同块设备一样,对字符设备的输入输出也是通过缓冲区来进行的。使用缓冲区有个额外
的好处,即以缓冲区为界,函数可分为高低两个层次。低层函数负责与实际设备交互,
而高层函数只与缓冲区打交道,只对缓冲区存取数据,这样可以蔽掉掉底层的许多细节。
对于字符缓冲区,有两个最重要结构,即cblock和clist。前者是缓冲区本身,后者则用作
字符链表(队列)的头结点。莱昂在第23章中详细介绍了这两个struct,并对字符链表的
增删操作作了详细的分析,请您务必仔细阅读第23章的说明。
系统共有NCLIST个缓冲区资源:
8146: struct cblock cfree[NCLIST]
空闲的缓冲区都被链接在cfreelist链表中。
Cinit函数使用头插法建立cfreelist链表,该函数比较令人费解的是第8240行的for循环——该条
语句的作用是,使每个实际使用的缓冲区的起始地址都能够被8整除。莱昂其实已经说明了这
样做的原因,即这样设置之后,可以通过地址模除8的方式来判断clist的头尾指针是否移出了
一个缓冲区边界。
getc和putc函数分别用来从缓冲区链表中get和put char。奇怪的是,unix v6选择使用汇编来实现
这两个函数——也许是出于性能的考虑吧。
getc函数有一个参数,即缓冲区链表的头结点地址。
(1)第936行将first ptr放入r2;
(2) 937行:如果头结点的first ptr为NULL(表示缓冲区已空),则跳到Label 9,返回-1;
(3) 938行:取first ptr指向的一个byte,放入r0(作为返回值);
注意,这里使用的movb指令以byte为单位进行操作,r2会后移1个byte,而不是word。
(4) 940行:将first ptr后移一个byte。
接下来的处理分为3种情况:
i.缓冲区链表已空;
此时,无需后续处理,直接从Label“3:”处返回即可。
ii.first ptr移出了当前缓冲区边界;
iii.first ptr仍在当前缓冲区内;
这两种情况需要后续处理,即调整first ptr,并释放用完的缓冲区。
putc函数有两个参数:
(1) 要放入缓冲区链表的char;
(2) 缓冲区链表头结点地址。
同getc不同,该函数在当前缓冲区链表满时,需要申请空闲缓冲区——当无空闲缓冲区可
用时,操作会失败,会通过Label 9返回非0数据。函数有几个重要的判断:
(1) 976: 当前缓冲链表是否为空(只有头结点);
(2) 978: 系统中是否还有空闲缓冲区;
(3) 984: 当前缓冲区是否已经写满
【注】:从putc函数可知,clist的last ptr指向的其实并不是缓冲链表内最后一个char,而是该
char的下一个byte。如果没有跨越缓冲区边界,则此byte可用于存放下一个char,否则,表示
缓冲链表已满,需要新申请一个缓冲区。
博客地址:http://blog.csdn.net/cszhao1980
博客专栏地址:http://blog.csdn.net/column/details/lions-unix.html
(莱昂氏unix源代码分析导读-49) 字符缓冲区的更多相关文章
- (莱昂氏unix源代码分析导读-50)LP11行式打印机
by cszhao1980 LP11有两个设备寄存器:状态寄存器(lpsr)和数据缓冲寄存器(lpbuf),可通过以下结构进行访问: 8812: #define LPADDR 0177514 8823 ...
- struts2请求过程源代码分析
struts2请求过程源代码分析 Struts2是Struts社区和WebWork社区的共同成果.我们甚至能够说,Struts2是WebWork的升级版.他採用的正是WebWork的核心,所以.Str ...
- 转:RTMPDump源代码分析
0: 主要函数调用分析 rtmpdump 是一个用来处理 RTMP 流媒体的开源工具包,支持 rtmp://, rtmpt://, rtmpe://, rtmpte://, and rtmps://. ...
- hostapd源代码分析(二):hostapd的工作机制
[转]hostapd源代码分析(二):hostapd的工作机制 原文链接:http://blog.csdn.net/qq_21949217/article/details/46004433 在我的上一 ...
- 【转载】linux环境下tcpdump源代码分析
linux环境下tcpdump源代码分析 原文时间 2013-10-11 13:13:02 CSDN博客 原文链接 http://blog.csdn.net/han_dawei/article/d ...
- Hadoop源代码分析
http://wenku.baidu.com/link?url=R-QoZXhc918qoO0BX6eXI9_uPU75whF62vFFUBIR-7c5XAYUVxDRX5Rs6QZR9hrBnUdM ...
- linux环境下tcpdump源代码分析
Linux 环境下tcpdump 源代码分析 韩大卫@吉林师范大学 tcpdump.c 是tcpdump 工具的main.c, 本文旨对tcpdump的框架有简单了解,只展示linux平台使用的一部分 ...
- 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系统匿名共 ...
随机推荐
- 每天收获一点点------Hadoop之HDFS基础入门
一.HDFS出现的背景 随着社会的进步,需要处理数据量越来越多,在一个操作系统管辖的范围存不下了,那么就分配到更多的操作系统管理的磁盘中,但是却不方便管理和维护—>因此,迫切需要一种系统来管理多 ...
- Theano学习笔记(三)——图结构
图结构(Graph Structures)这是理解Theano该基金会的内部运作. Theano编程的核心是用符号占位符把数学关系表示出来. 图结构的组成部分 如图实现了这段代码: importthe ...
- JavaScript之事件处理详解
一.事件传播机制 客户端JavaScript程序(就是浏览器啦)采用了异步事件驱动编程模型.当文档.浏览器.元素或与之相关的对象发生某些有趣的事情时,Web浏览器就会产生事件(event).如果Jav ...
- C++内存分配和拷贝构造函数写研究
昨晚参加笔试,开错题,有印象中的概念,但目前尚不清楚是怎么回事,什么原理,导致错误的话题.现在总结. 一.C++写内存分配研究 问题考察例如以下,请先不要看答案,看看你是否能做对,呵呵: waterm ...
- java Json字符串转List<Map>类型
//相关包 import com.google.gson.Gson; import com.google.gson.JsonElement; import com.google.gson.JsonPa ...
- JQUERY prop与attr差额
1. 1-9-1之前和之后之间的差 <html> <script src="Js/jquery-1.9.0.js" type="text/javasc ...
- Log4j2 简明教程
一.概述 log4j2官方文档内容非常多,要一次性了解全部是不可能的.正确的步骤应当是先了解最常见的配置,当发现原有知识无法解决问题,再重新查看文档看有没有合适的配置.下面将从文件结构入手,再到简单的 ...
- JavaScript中获取当前项目的绝对路径
近期在做JavaWeb项目相关的东西,差点儿每天都遇到非常多问题,主要是由于自己对JavaWeb方面的知识不是非常清楚,尽量把自己在项目中遇到的问题都记录下来,方便以后查阅. 在我们的项目中有这种须要 ...
- 数据持久化之SP的优化—送工具类
第一点:sp存储的是键值对 getSharedPreferences 第一个參数是你保存文件的名字,第个是保存的模式一般能够默觉得0 先看普通 使用SP 存储String类型字符串吧 SharedPr ...
- js 上一天、下一天
function Previousday() { var curDate = new Date(CreateDate_Temp); * * * ); var strDate = preDate.get ...