1. 介绍

本文介绍了TTY打开、TTY读和TTY写操作的函数跟踪过程

2. 示例

下面是一个简单的Linux TTY打开和读写过程

#include <termios.h>
#include <unistd.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h> int main()
{
struct termios toptions;
int fd, falgs; flags = O_RDWR | O_NOCTTY | O_SYNC | O_NDELAY;
fd = open("/dev/ttyS0", flags); tcflush(fd, TCIOFLUSH);
memset(&toptions, , sizeof(toptions));
tcgetattr(fd, &toptions);
cfmakeraw(&toptions);

tcgetattr(fd, &toptions);
cfsetospeed(&toptions, B115200);
cfsetispeed(&toptions, B115200); tcgetattr(fd, &toptions);
toptions.c_cflag = B115200 | CRTSCTS | CS8 | CLOCAL | CREAD | INPCK;
toptions.c_cc[VMIN] = ;
toptions.c_cc[VTIME] = ; tcflush(fd, TCIOFLUSH);
tcsetattr(fd, TCSANOW, &toptions); uint8_t data[] = { 0x01, 0x09, 0x10, 0x00 };
write(fd, data, ); memset(read_data, , );
read(fd, data, ); return ;
}

3. 打开

open会调用tty_open, 其分析如下:

tty_open
tty_alloc_file
/* 分配tty_file_private并赋值给file::private_data */
tty_open_current_tty
/* 如果当前设备是/dev/tty则尝试重新打开, 通常返回NULL */
tty_open_by_driver
/* 如果上面的函数返回NULL通过查找tty驱动打开tty设备 */
tty_lookup_driver
/* 根据设备号查找对应的tty驱动 */
tty_driver_lookup_tty
/* 根据tty_driver::tty_operations::lookup/tty_driver::tty_struct查找tty_struct是否存在 */
tty_reopen
/* 如果tty_struct已分配, 则打开该tty */
tty_ldisc_reinit
/* 初始化tty设备的线路规程 */
tty_ldisc_get
/* 根据类别(默认为N_TTY)获取注册的对应线路规程, 并分配对应tty_ldisc实例 */
tty_set_termios_ldisc
/* 设置该tty的线路规程 */
tty_ldisc_open
/* 调用tty_ldisc::tty_ldisc_ops::open, 对于N_TTY为n_tty_open */
tty_init_dev
/* 如果tty_struct没有分配则进行分配(!!!uart就是如此!!!) */
alloc_tty_struct
/* 分配和初始化一个tty_struct, 包括driver, ops, index等 */
tty_ldisc_init
/* 初始化tty_struct的线路规程 */
tty_ldisc_get
/* 获取N_TTY对应线路规程, 并分配对应tty_ldisc实例 */
tty_struct::tty_operations = tty_driver::tty_operations
/* 设置tty_struct的函数操作集 */
tty_driver_install_tty
/* 通过tty_driver::tty_operations::install或者tty_standard_install为tty设备安装入口 */
tty_ldisc_setup
/* 打开线路规程 */
tty_ldisc_open
/* 调用tty_ldisc::tty_ldisc_ops::open, 对于N_TTY为n_tty_open */
tty_add_file
/* 关联文件到tty_struce */
tty_struct::tty_operations::open
/* 同tty_driver::tty_operations::open, 对于串口即uart_open */

uar_open分析如下

uart_open
tty_port_open
uart_port_activate
/* 即tty_port::tty_port_operations::activate */
uart_startup
uart_port_startup
uart_change_pm
/* 调用uart_port::uart_ops::pm */
get_zeroed_page
/* 分配一页并分配给uart_state.xmit.buf */
uart_circ_clear
/* 初始化TX环形缓冲区头和尾指针 */
uart_port::uart_ops::startup
uart_change_speed

4. 写入

write会调用tty_write, 其分析如下:

tty_write
file_tty
/* 通过file::tty_file_private::tty_struct获取对应的tty_struct */
do_tty_write
/* 调用tty_struct::tty_ldisc::tty_ldisc_ops::write */
n_tty_write
/* 对于类型N_TTY的线路规程, write为n_tty_write */
process_echoes
/* 处理所有待处理的echoed字符 */
__process_echoes
/* 写入待处理的echo字符 */
tty_write_room
/* 调用tty_struct::tty_operations::write_room */
echo_buf
tty_put_char
/*
* 调用tty_struct::tty_operations::put_char
* 或调用tty_struct::tty_operations::write
*/
do_output_char
/* 调用tty_struct::tty_operations::write */
tty_struct::tty_operations::flush_chars
/* 对于uart为uart_flush_buffer */
process_output_block
process_output
tty_struct::tty_operations::flush_chars
tty_struct::tty_operations::write
/* 对于uart为uart_write */

uart_write分析如下

uart_write
uart_state::circ_buf
/* 获取环形缓冲区 */
memcpy
/* 将待写数据拷贝至环形缓冲区 */
__uart_start
uart_port::uart_ops::wake_peer
uart_port::uart_ops::start_tx

5. 读取

read会调用tty_read, 其分析如下:

tty_read
tty_struct::tty_ldisc::tty_ldisc_ops::read
/* 对于类型N_TTY的线路规程, 为n_tty_read */
n_tty_read
tty_buffer_flush_work
flush_work
flush_to_ldisc
/* 将数据从tty缓冲区刷新到线路规程 */
receive_buf
tty_ldisc_receive_buf
/* 调用tty_struct::tty_ldisc::tty_ldisc_ops::receive_buf2 */
n_tty_receive_buf2
n_tty_receive_buf_common
__receive_buf
n_tty_receive_buf_real_raw
canon_copy_from_read_buf
/* ICANON ON */
copy_from_read_buf
/* ICANON OFF */
copy_to_user
/* 将数据拷贝至用户空间 */

Linux TTY函数跟踪的更多相关文章

  1. Linux TTY驱动--Serial Core层【转】

    转自:http://blog.csdn.net/sharecode/article/details/9197567 版权声明:本文为博主原创文章,未经博主允许不得转载. 接上一节: Linux TTY ...

  2. linux select函数详解

    linux select函数详解 在Linux中,我们可以使用select函数实现I/O端口的复用,传递给 select函数的参数会告诉内核: •我们所关心的文件描述符 •对每个描述符,我们所关心的状 ...

  3. 20135239 益西拉姆 linux内核分析 跟踪分析Linux内核的启动过程

    回顾 1.中断上下文的切换——保存现场&恢复现场 本节主要课程内容 Linux内核源代码简介 1.打开内核源代码页面 arch/目录:支持不同CPU的源代码:其中的X86是重点 init/目录 ...

  4. 动态替换Linux核心函数的原理和实现

    转载:https://www.ibm.com/developerworks/cn/linux/l-knldebug/ 动态替换Linux核心函数的原理和实现 在调试Linux核心模块时,有时需要能够实 ...

  5. Linux TTY介绍

    1. TTY介绍 TTY(TeleType)指Linux中的一类终端(Terminal)设备, 是一种字符设备 在Linux中, tty可分为如下几类- 串行端口终端(serial port term ...

  6. Linux Clone函数

    Linux Clone函数 之前某一次有过一次面试,问了内核中是怎么创建命名空间的? 下面就来扒一扒clone的精髓,以及如何通过它创建命名空间. 目录 Linux Clone函数 使用clone创建 ...

  7. Xdebug文档(四)函数跟踪

    Xdebug能让你把所有函数调用,包括参数和返回值以不同的格式记录到文件中. 这些号称“函数跟踪”功能能帮助你面对一个新应用程序,亦或者在程序运行时你想弄清楚它在做什么.函数跟踪功能可以选择性地显示函 ...

  8. Linux open函数

    Linux open函数 open 函数用于打开和创建文件.以下是 open 函数的简单描述 #include <fcntl.h> int open(const char *pathnam ...

  9. Linux TTY框架【转】

    本文转载自:http://ju.outofmemory.cn/entry/281168 1. 前言 由于串口的缘故,TTY是Linux系统中最普遍的一类设备,稍微了解Linux系统的同学,对它都不陌生 ...

随机推荐

  1. 海康—SADP激活(设备网络搜索)

    海康sadp搜索工具(SADPTool)用于从网络上搜索同一网段内的所有在线设备.可以修改设备的缺省密码,修改网络IP地址及端口号 ,子网掩码及网关地址,IPV6地址网关地址,HTTP端口号和设备序列 ...

  2. 关于千里马招标网知道创宇反爬虫521状态码的解决方案(python代码模拟js生成cookie _clearence值)

    一.问题发现 近期我在做代理池的时候,发现了一种以前没有见过的反爬虫机制.当我用常规的requests.get(url)方法对目标网页进行爬取时,其返回的状态码(status_code)为521,这是 ...

  3. Spring AOP的常用方法

    转  https://blog.csdn.net/u014745069/article/details/84887765

  4. Java生成二进制文件与Postman以二进制流的形式发送请求

    业务描述: 模拟终端(智能家居)发送HTTP POST请求,请求参数为二进制流:而且,二进制流是加密后的数据,因此调试分两步: 1.Java代码生成加密后数据,并保存为二进制流文件 (电脑上的图片就是 ...

  5. 第十届蓝桥杯大赛-特别数的和-C++

    解法一(暴力获取): #include<stdio.h> #include<stdlib.h> int main(void) { int n; ; ; printf(" ...

  6. Go基础编程实践(五)—— 错误和日志

    自定义错误类型 Go中可以使用errors.New()创建错误信息,也可以通过创建自定义错误类型来满足需求.error是一个接口类型,所有实现该接口的类型都可以当作一个错误类型. // error类型 ...

  7. MacBook Pro配置汇编开发环境

    配置开发环境 方法一: 打开命令行,输入指令which nasm查看nasm的安装路径,Mac系统默认安装了nasm.一般默认返回的路径是/usr/bin/nasm 接着输入指令alias nasm= ...

  8. 如何监视 WPF 中的所有窗口,在所有窗口中订阅事件或者附加 UI

    原文:如何监视 WPF 中的所有窗口,在所有窗口中订阅事件或者附加 UI 由于 WPF 路由事件(主要是隧道和冒泡)的存在,我们很容易能够通过只监听窗口中的某些事件使得整个窗口中所有控件发生的事件都被 ...

  9. php中的for循环和js中的for循环

    php中的for循环 循环100个0 for ($i=0;$i<=100;$i++){ $pnums.='0'.","; } js中的for循环,循环31个相同的数.循环日期 ...

  10. SpringBoot的入门程序

    1. 创建一个springboot工程 可以参考springboot入门程序 2. 创建一个实体类 @Data //想相当于@Setter.@Getter和@ToString替代了setter.get ...