line discipline(LDISC) 线路规程,是linux和类unix系统终端子系统的一个软件驱动层。终端子系统从上到下可划分为三层:
- 顶层tty core驱动层提供字符设备接口(因为所有的终端设备都是字符设备);
- 最底层是tty driver层用来和硬件进行通讯,实现tty_operations供tty core和 LDISC层调用;
- 中间层line discipline实现终端输入输出数据处理的策略。
驱动的工作是以硬件能够理解的方式格式化发送给它的数据, 并且从硬件接收数据.
也就是说,驱动的作用是接受硬件数据或者发送硬件能够理解的数据至硬件设备,并且承担了配置硬件寄存器的任务。
tty 线路规程的工作是以特殊的方式格式化从一个用户或者硬件收到的数据. 这种格式化常常采用一个协议转换的形式, 例如 PPP 和 Bluetooth。
也就是说,线路规程用于格式化在驱动层以上的数据。格式化的作用是让整个设备具有额外的功能。如PPP作为线路规程的时候,将具有鉴权、压缩等功能。
line discipline把tty core和tty driver粘合在一起,策略的分离使得tty core和tty driver不需要关注数据语法处理,tty driver可以被相同的硬件复用,而只需更改line discipline。
linux终端设备缺省的线路规程是N_TTY,它从tty driver和应用程序接收数据,按照终端设置处理数据。对于输入数据,它处理特殊的中断字符(比如Control-C),删除字符(backspace, delete)等等;对于输出数据,它用CR/LF序列替换LF字符。当uart port用做普通串口时,使用N_TTY线路规程。
当uart port设备用做serial modem 的internet拨号连接时,使用PPP线路规程处理数据;ppp LDISC把从uart core来的串口数据组装为PPP输入packet,然后分发给网络协议栈;ppp LDISC把从网络协议栈发送来的数据拆包发送给uart port。
ldisc工作于tty core和tty driver之间,ldisc需要为二者提供如下接口函数:
TTY core端接口:
open()
close()
hangup()
write()
flush_buffer()
chars_in_buffer()
set_termios()
read()
poll()
ioctl()
compat_ioctl()
TTY driver端接口:
receive_buf()
write_wakeup()
dcd_change()