linux驱动(第二十课 TTY)

console=ttySAC0,115200
这是bootargs 里最常见的一句。它被用来选择从哪个设备输出内核、驱动的printk信息。

这究竟是什么意思?
串行端口终端(/dev/ttySn)
计算机把每个串行端口都看作是一个字符设备。这些串行端口所对应的设备名称是/dev/ttyS0,/dev/ttyS1等,设备号分别是(4,0), (4,1)等.
PC上的串口一般是ttySn,板子上Linux的串口一般叫做ttySACn.

控制终端(/dev/tty)
如果当前进程有控制终端(Controlling Terminal)的话,那么/dev/tty就是当前进程的控制终端的设备特殊文件。对于你登录的shell,/dev/tty就是你使用的终端,设备号是(5,0)。不同的程序打开这个设备文件可能指向的终端不同。

控制台终端(/dev/ttyn, /dev/console)
tty1–tty6等 称为虚拟终端,而tty0则是当前所使用虚拟终端的一个别名current tty,系统所产生的信息会发送到该终端上(这时也叫控制台终端)。
/dev/console即控制台,是与操作系统交互的设备,系统将一些信息直接输出到控制台上。

Linux内核中,有个console设备对象,它会关联到某个具体的设备,例如串口或者FrameBuffer,当printk有消息打印打印到console中时,console会调用关联的具体设备的fops.write,将信息输出到具体设备上。
如果是串口,那么就会在串口上出现发送的波形,如果是framebuffer,那么就会在LCD上显示出来。

console=ttySAC0,就是将板子上的具体设备,/dev/ttySAC0,关联到console设备。
当console有数据输出时,会调用ttySAC0.fops.write发送。

那么这个ttySAC0又是怎么来的呢?
这就涉及到linux的TTY子系统了。
TT驱动架构,也是一个分层的架构,包含了TTYCORE,TTYROUTINE,TTYDRIVER等。
TTYCORE在tty_io.c文件中实现,被实现为一个CDEV,它对用户程序提供CDEV的访问机制,即fops,对下层,又需要调用tty_driver中的函数接口。
这样,tty设备驱动的工作,就转化为填充tty_driver实例,并编写对应的tty_operations的成员函数。

虽然UART驱动完全可以遵循tty_driver的架构进行设计,但是linux还是在
drivers/tty/serial/serial_core.c中,专门实现了UART的通用TTY层,即串口核心层。这样,TTYDRIVER使用UARTCORE提供的服务,而UARTCORE则向下层提供定义好的驱动接口,由具体的串口驱动来提供服务。

串口核心层定义了uart_driver结构体,及其操作集 uart_ops。
串口核心层会调用uart_ops中的函数接口,向上层提供服务。

struct uart_driver{
	struct tty_driver * tty_driver;
	
	const char* driver_name;
	const char* dev_name;
	
	int major;
	int minor;
	
	struct console* cons;
};

struct uart_ops{
	void (*start_tx)(struct uart_port*);
	void (*flush_buffer)(*struct uart_port*);
	void (*config_port)(*struct uart_port*, int );
	...
};



内核中,用uart_port来表示一个UART设备,uart_ops被关联到UART_PORT上,作为UART的操作集。
当uart_driver中的函数接口被调用时,会传入一个uart_port的对象,那么就可以通过uart_port.ops来找到所需要的操作集了。

内核提供了API
uart_register_driver()
用来注册一个UART的驱动。

这样,一个具体的UART设备的驱动编写,就变成填充uart_driver,注册uart_driver,并编写uart_ops了。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
LINUX设备驱动第三版_ 前言 第一章 设备驱动程序简介 设备驱动程序的作用 内核功能划分 设备和模块的分类 安全问题 版本编号 许可证条款 加入内核开发社团 本书概要 第二章 构造和运行模块 设置测试系统 Hello World模块 核心模块与应用程序的对比 编译和装载 内核符号表 预备知识 初始化和关闭 模块参数 在用户空间编写驱动程序 快速参考 第三章 字符设备驱动程序 scull的设计 主设备号和次设备号 一些重要的数据结构 字符设备的注册 open和release scull的内存使用 read和write 试试新设备 快速参考 第四章 调试技术 内核中的调试支持 通过打印调试 通过查询调试 通过监视调试 调试系统故障 调试器和相关工具 第五章 并发和竞态 scull的缺陷 并发及其管理 信号量和互斥体 completion 自旋锁 锁陷阱 除了锁之外的办法 快速参考 第六章 高级字符驱动程序操作 ioctl 阻塞型I/O poll和select 异步通知 定位设备 设备文件的访问控制 快速参考 第七章 时间、延迟及延缓操作 度量时间差 获取当前时间 延迟执行 内核定时器 tasklet 工作队列 快速参考 第八章 分配内存 kmalloc函数的内幕 后备高速缓存 get_free_page和相关函数 vmalloc及其辅助函数 per-CPU变量 获取大的缓冲区 快速参考 第九章 与硬件通信 I/O端口和I/O内存 使用I/O端口 I/O端口示例 使用I/O内存 快速参考 第十章 中断处理 准备并口 安装中断处理例程 实现中断处理例程 顶半部和底半部 中断共享 中断驱动的I/O 快速参考 第十一章 内核的数据类型 使用标准C语言类型 为数据项分配确定的空间大小 接口特定的类型 其他有关移植性的问题 链表 快速参考 第十二章 PCI驱动程序 PCI接口 ISA回顾 PC/104和PC/104+ 其他的PC总线 SBus NuBus 外部总线 快速参考 第十三章 USB驱动程序 USB设备基础 USB和Sysfs USB urb 编写USB驱动程序 不使用urb的USB传输 快速参考 第十四章 Linux设备模型 kobject、kset和子系统 低层sysfs操作 热插拔事件的产生 总线、设备和驱动程序 类 各环节的整合 热插拔 处理固件 快速索引 第十五章 内存映射和DMA Linux的内存管理 mmap设备操作 执行直接I/O访问 直接内存访问 快速参考 第十六章 块设备驱动程序 注册 块设备操作 请求处理 其他一些细节 快速参考 第十七章 网络驱动程序 snull设计 连接到内核 net_device结构细节 打开和关闭 数据包传输 数据包的接收 中断处理例程 不使用接收中断 链路状态的改变 套接字缓冲区 MAC 地址解析 定制 ioctl 命令 统计信息 组播 其他知识点详解 快速参考 第十八章 TTY驱动程序 小型TTY驱动程序 tty_driver函数指针 TTY线路设置 ioctls proc和sysfs对TTY设备的处理 tty_driver结构详解 tty_operations结构详解 tty_struct结构详解 快速参考 参考书目 9112405-1_o.jpg (85.53 KB, 下载次数: 50)

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值