《LINUX设备驱动开发详解》作者:华清远见第14章Linux终端设备驱动LinuxLinux314.114.2Linuxtty_driver14.314.514.2Linux/open()close()ttyLinuxtty14.6Linuxtty14.7tty14.814.614.7ttyS3C2410UART专业始于专注卓识源于远见 ‐ 2 ‐终端设备在Linux系统中,终端是一种字符型设备,它有多种类型,通常使用tty来简称各种类型的终端设备。tty是Teletype的缩写,Teletype是最早出现的一种终端设备,很像电传打字机,是由Teletype公司生产的。Linux系统中包含如下几类终端设备。1.串行端口终端(/dev/ttySn) 串行端口终端(SerialPortTerminal)是使用计算机串行端口连接的终端设备。计算机把每个串行端口都看作是一个字符设备。这些串行端口所对应的设备名称是/dev/ttyS0(或/dev/tts/0)、/dev/ttyS1(或/dev/tts/1)等,设备号分别是(4,0)、(4,1)等。在命令行上把标准输出重定向到端口对应的设备文件名上就可以通过该端口发送数据,例如,在命令行提示符下输入“echotest/dev/ttyS1”会把单词“test”发送到连接在ttyS1端口的设备上。2.伪终端(/dev/pty/) 伪终端(PseudoTerminal)是成对的逻辑终端设备,并存在成对的设备文件,如/dev/ptyp3和/dev/ttyp3,它们与实际物理设备并不直接相关。如果一个程序把ttyp3看作是一个串行端口设备,则它对该端口的读/写操作会反映在该逻辑终端设备对应的ttyp3上,而ttyp3则是另一个程序用于读写操作的逻辑设备。这样,两个程序就可以通过这种逻辑设备进行通信,使用ttyp3的程序会认为自己正在与一个串行端口进行通信。以telnet为例,如果某人在使用telnet程序连接到Linux系统,则telnet程序就可能会开始连接到设备ptyp2上,而此时一个getty程序会运行在对应的ttyp2端口上。当telnet从远端获取了一个字符时,该字符就会通过ptyp2、ttyp2传递给getty程序,而getty程序则会通过ttyp2、ptyp2和telnet程序返回“login:”字符串信息。这样,登录程序与telnet程序就通过伪终端进行通信。通过使用适当的软件,可以把两个或多个伪终端设备连接到同一个物理串行端口上。3.控制台终端(/dev/ttyn,/dev/console) 如果当前进程有控制终端(ControllingTerminal),那么/dev/tty就是当前进程的控制终端的设备特殊文件。可以使用命令“ps–ax”来查看进程与哪个控制终端相连,使用命令“tty”可以查看它具体对应哪个实际终端设备。/dev/tty有些类似于到实际所使用终端设备的一个连接。在UNIX系统中,计算机显示器通常被称为控制台终端(Console)。它仿真了类型为Linux的一种终端(TERM=Linux),并且有一些设备特殊文件与之相关联:tty0、tty1、tty2等。当用户在控制台上登录时,使用的是tty1。按[Alt+F1]~[Alt+F6]组合键时,我们就可以切换到tty2、tty3等。tty1~tty6等称为虚拟终端,而tty0则是当前所使用虚拟终端的一个别名,系统所产生的信息会发送到该终端上。因此不管当前正在使用哪个虚拟终端,系统信息都会发送到控制台终端上。用户可以登录到不同的虚拟终端上去,因而可以让系统同时有几个不同的会话期存在。只有系统或超级用户root可以向/dev/tty0进行写操作。在Linux系统中,可以在系统启动命令行里指定当前的输出终端,格式如下:console=device,optionsdevice指代的是终端设备,可以是tty0(前台的虚拟终端)、ttyX(第X个虚拟终端)、ttySX(第X个串口)、lp0(第一个并口)等。options指代对device进行的设置,它取决于具体的设备驱动。对于串口设备,参数用来定义为如下。专业始于专注卓识源于远见 ‐ 3 ‐波特率、校验位、位数,格式为BBBBPN,其中BBBB表示波特率,P表示校验(n/o/e),N表示位数,默认options是9600n8。用户可以在内核命令行中同时设定多个终端,这样输出将会在所有的终端上显示,而当用户调用open()打开/dev/console时,打开的将是设定的最后一个终端。例如:console=ttyS1,9600console=tty0定义了两个终端,而调用open()打开/dev/console时,将使用虚拟终端tty0。但是内核消息会在tty0VGA虚拟终端和串口ttyS1上同时显示。通过查看/proc/tty/drivers文件可以获知什么类型的tty设备存在以及什么驱动被加载到内核,这个文件包括一个当前存在的不同tty驱动的列表,包括驱动名、默认的节点名、驱动的主编号、这个驱动使用的次编号范围以及tty驱动的类型。例如,下面所示为一个/proc/tty/drivers文件的例子。终端设备驱动结构Linux内核中tty的层次结构如图14.1所示,包含tty核心、tty线路规程和tty驱动,tty线路规程的工作是以特殊的方式格式化从一个用户或者硬件收到的数据,这种格式化常常采用一个协议转换的形式,例如PPP和Bluetooth。tty设备发送数据的流程为:tty核心从一个用户获取将要发送给一个tty设备的数据,tty核心将数据传递给tty线路规程驱动,接着数据被传递到tty驱动,tty驱动将数据转换为可以发送给硬件的格式。接收数据的流程为:从tty硬件接收到的数据向上交给tty驱动,进入tty线路规程驱动,再进入tty核心,在这里它被一个用户获取。尽管大多数时候tty核心和tty之间的数据传输会经历tty线路规程的转换,但是tty驱动与tty核心之间也可以直接传输数据。图14.2显示了与tty相关的主要源文件及数据的流向。tty_io.c定义了tty设备通用的file_operations结构体并实现了接口函数tty_register_driver()用于注册tty设备,它会利用fs/char_dev.c提供的接口函数注册字符设备,与具体设备对应的tty驱动将实现tty_driver结构体中的成员函数。同时tty_io.c也提供了tty_register_ldisc()接口函数用于注册线路规程,n_tty.c文件则实现了tty_disc结构体中的成员。14.1tty专业始于专注卓识源于远见 ‐ 4 ‐14.2tty从图14.2可以看出,特定tty设备驱动的主体工作是填充tty_driver结构体中的成员,实现其中的成员函数,tty_driver结构体的定义如代码清单14.1所示。代码清单14.1tty_driver结构体1structtty_driver2{3intmagic;4structcdevcdev;/*对应的字符设备cdev*/5structmodule*owner;/*这个驱动的模块拥有者*/6constchar*driver_name;7constchar*devfs_name;8constchar*name;/*设备名*/9intname_base;/*offsetofprintedname*/10intmajor;/*主设备号*/11intminor_start;/*开始次设备号*/12intminor_num;/*设备数量*/13intnum;/*被分配的设备数量*/14shorttype;/*tty驱动的类型*/15shortsubtype;/*tty驱动的子类型*/16structtermiosinit_termios;/*初始线路设置*/17intflags;/*tty驱动标志*/18intrefcount;/*引用计数(针对可加载的tty驱动)*/19structproc_dir_entry*proc_entry;/*/proc文件系统入口*/20structtty_driver*other;/*仅对PTY驱动有意义*/21...22/*接口函数*/23int(*open)(structtty_struct*tty,structfile*filp);24void(*close)(structtty_struct*tty,structfile*filp);25int(*write)(structtty_struct*tty,constunsignedchar*buf,intcount);26void(*put_char)(structtty_struct*tty,unsignedcharch);27void(*flush_chars)(structtty_struct*tty);28int(*write_room)(structtty_struct*tty);29int(*chars_in_buffer)(structtty_struct*tty);30int(*ioctl)(structtty_struct*tty,structfile*file,unsignedintcmd,31unsignedlongarg);32void(*set_termios)(structtty_struct*tty,structtermios*old);33void(*throttle)(structtty_struct*tty);34void(*unthrottle)(structtty_struct*tty);35void(*stop)(structtty_struct*tty);36void(*start)(structtty_struct*tty);37void(*hangup)(structtty_struct*tty);38void(*break_ctl)(structtty_struct*tty,intstate);39void(*flush_buffer)(structtty_struct*tty);40void(*set_ldisc)(structtty_struct*tty);41void(*wait_until_sent)(structtty_struct*tty,inttimeout);42void(*send_xchar)(structtty_struct*tty,charch);43int(*read_proc)(char*page,char**start,off_toff,intcount,int*eof,专业始于专注卓识源于远见 ‐ 5 ‐44void*data);45int(*write_proc)(structfile*file,constchar__user*buffer,unsignedlong46count,void*data);47int(*tiocmget)(structtty_struct*tty,structfile*file);48int(*tiocmset)(structtty_struct*tty,structfile*file,unsignedintset,49unsignedintclear);5051structlist_headtty_drivers;52};tty_driver结构体中的magic表示给这个结构体的“幻数”,设为TTY_DRIVER_MAGIC,在alloc_tty_driver()函数中被初始化。name与driver_name的不同在于后者表示驱动的名字,用在/proc/tty和sysfs中,而前者表示驱动的设备节点名。type与subtype描述tty驱动的类型和子类型,subtype的值依赖于type,type成员的可能值为TTY_DRIVER_TYPE_SYSTEM(由tty子系统内部使用,subtype应当设为SYSTEM_TYPE_