Linux驱动开发笔记(十一)tty子系统及其驱动

文章目录

  • 前言
  • 一、串口驱动框架
    • 1.1 核心数据结构
    • 1.2 数据处理流程
  • 二、驱动编写
    • 1. 设备树的修改
    • 2. 相关API函数
    • 3. 驱动框架
    • 4. 具体功能的实现
      • 4.1 出入口函数的编写
      • 4.2 读写函数


前言

  之前已经讲过应用层的应用,接下来我们继续进行驱动的学习。其实实际上我们很少主动进行串口的驱动编写,通常情况下只需要进行应用层的应用就可以了,网络上相关的驱动内容介绍也较少,这里仅作了解并简单了解一下架构即可。


一、串口驱动框架

  串口驱动没有什么主机端和设备端之分,就只有一个串口驱动,而且这个驱动也已经由厂商已经编写好了,我们真正要做的就是在设备树中添加所要使用的串口节点信息。当系统启动以后串口驱动和设备匹配成功,相应的串口就会被驱动起来,生成/dev/ttymxcX(X=0….n)文件。
  这部分内容还是比较多的,笔者发现这个文章讲述的还是很详细的,感兴趣可以自行查阅,这里不再赘述:框架详细介绍

1.1 核心数据结构

  对于串口驱动来说,实际上重要的结构体只有uart_drive和uart_port,这里可能根据不同的厂商所采用的名称不大一样,但内容差不多,比如在rk3568中,官方提供的uart驱动程序位8250通用串口程序,其为uart_port结构体提供了扩展后的uart_8250_port,包含8250 UART特有的属性和操作。
  这部分内容保存于内核中的下列文件中,感兴趣可以自行查阅:

  •  serial_core.c:包含UART核心层的实现。
  •  8250.c:包含8250 UART特定操作的实现。
  •  8250_port.c:包含8250 UART端口的具体实现。

  uart_driver结构体用于描述一个UART驱动程序,它通常包含一些基础的信息和操作函数。

struct uart_driver {struct module *owner;               // 设备驱动模块的所有者const char *driver_name;            // 驱动程序的名称const char *dev_name;               // 设备名称int major;                          // 设备主设备号int minor;                          // 设备次设备号int nr;                             // 该驱动程序支持的设备数量struct console *cons;               // 控制台相关信息struct uart_ops *ops;               // UART操作函数集合struct uart_state *state;struct tty_driver *tty_driver;
};

  uart_8250_port结构体用于描述8250 UART端口的具体信息和操作方法。

struct uart_8250_port {struct uart_port port;              // 通用UART端口结构体struct timer_list timer;            // 定时器用于处理超时unsigned int capabilities;          // 硬件能力unsigned int mcr;                   // 调制解调器控制寄存器unsigned int lsr;                   // 线路状态寄存器unsigned int msr;                   // 调制解调器状态寄存器unsigned int icr;                   // 中断控制寄存器unsigned int lcr;                   // 线路控制寄存器unsigned int fcr;                   // FIFO控制寄存器unsigned int ier;                   // 中断使能寄存器unsigned char mcr_mask;             // 调制解调器控制寄存器掩码unsigned char mcr_force;            // 调制解调器控制寄存器强制值unsigned char lsr_break_flag;       // 断开标志unsigned char bugs;                 // 硬件错误标志unsigned int tx_loadsz;             // 发送FIFO加载大小unsigned int acr;                   // 额外控制寄存器unsigned int ier_mask;              // 中断使能寄存器掩码unsigned int ier_force;             // 中断使能寄存器强制值unsigned int lsr_mask;              // 线路状态寄存器掩码unsigned int lsr_break_flag_mask;   // 断开标志掩码
};//通用结构体,但比较鸡肋
struct uart_port {spinlock_t lock; /* port lock */unsigned long iobase; /* in/out[bwl] */unsigned char __iomem *membase; /* read/write[bwl] */
......const struct uart_ops *ops;unsigned int custom_divisor;unsigned int line; /* port index */unsigned int minor;resource_size_t mapbase; /* for ioremap */resource_size_t mapsize;struct device *dev; /* parent device */

  这里补充一下在uart_driver提到的实现控制台打印功能必须要注册的结构体console和将uart_port与对应的circ_buf联系起来的uart_state结构体。

struct console {char name[16];void(*write)(struct console *const char *, unsigined);int (*read)(struct console *, char *, unsigned);struct tty_driver *(struct console *,int*);void (*unblank)(void);int  (*setup)(struct console *, char *);int  (*early_setup)(void);short  flags;short  index; /*用来指定该console使用哪一个uart port (对应的uart_port中的line),如果为-1,kernel会自动选择第一个uart port*/int   cflag;void  *data;struct   console *next;
};/*uart_state有两个成员在底层串口驱动会用到,即xmit和port。
用户空间程序通过串口发送数据时,上层驱动将用户数据保存在xmit;而串口发送中断处理函数就是通过xmit获取到用户数据并将它们发送出去。
串口接收中断处理函数需要通过port将接收到的数据传递给线路规程层。
*/
struct uart_state {struct  tty_port  port;enum uart_pm_state   pm_state;struct circ_buf     xmit;struct uart_port     *uart_port; /*对应于一个串口设备*/
}

  uart_ops结构体几乎涵盖了驱动可对串口的所有操作:

 struct uart_ops {unsigned int    (*tx_empty)(struct uart_port *);void            (*set_mctrl)(struct uart_port *, unsigned int mctrl);unsigned int    (*get_mctrl)(struct uart_port *);void            (*stop_tx)(struct uart_port *);void            (*start_tx)(struct uart_port *);void            (*throttle)(struct uart_port *);void            (*unthrottle)(struct uart_port *);void            (*send_xchar)(struct uart_port *, char ch);void            (*stop_rx)(struct uart_port *);void            (*enable_ms)(struct uart_port *);void            (*break_ctl)(struct uart_port *, int ctl);int             (*startup)(struct uart_port *);void            (*shutdown)(struct uart_port *);void            (*flush_buffer)(struct uart_port *);void            (*set_termios)(struct uart_port *, struct ktermios *new,struct ktermios *old);void            (*set_ldisc)(struct uart_port *, int new);void            (*pm)(struct uart_port *, unsigned int state,unsigned int oldstate);int             (*set_wake)(struct uart_port *, unsigned int state);/** Return a string describing the type of the port*/const char      *(*type)(struct uart_port *);/** Release IO and memory resources used by the port.* This includes iounmap if necessary.*/void            (*release_port)(struct uart_port *);/** Request IO and memory resources used by the port.* This includes iomapping the port if necessary.*/int             (*request_port)(struct uart_port *);void            (*config_port)(struct uart_port *, int);int             (*verify_port)(struct uart_port *, struct serial_struct *);int             (*ioctl)(struct uart_port *, unsigned int, unsigned long);
#ifdef CONFIG_CONSOLE_POLLint             (*poll_init)(struct uart_port *);void            (*poll_put_char)(struct uart_port *, unsigned char);int             (*poll_get_char)(struct uart_port *);
#endif
};

1.2 数据处理流程

  相较于前面学习的spi子系统和iic子系统,uart驱动通常情况不需要根据协议的时序额外编写新的传输函数,这里可以直接提供write函数和read函数实现数据的互传。
在这里插入图片描述
  这部分的内容我们在应用层实验的时候已经详细介绍过了,感兴趣可以回顾一下。

二、驱动编写

  我们这里采用的8250通用驱动,对于不同驱动其函数名称和入口参数可能有所不同,这里我们简单学习梳理一下框架即可,通常来说我们是只需要进行应用层的编写。

1. 设备树的修改

代码如下(示例):

uart3: serial@fe670000 {compatible = "rockchip,rk3568-uart", "snps,dw-apb-uart";reg = <0x0 0xfe670000 0x0 0x100>;interrupts = <GIC_SPI 119 IRQ_TYPE_LEVEL_HIGH>;clocks = <&cru SCLK_UART3>, <&cru PCLK_UART3>;clock-names = "baudclk", "apb_pclk";reg-shift = <2>;reg-io-width = <4>;dmas = <&dmac0 6>, <&dmac0 7>;pinctrl-names = "default";pinctrl-0 = <&uart3m0_xfer>;status = "disabled";};//uart3的复用
uart3 {/omit-if-no-ref/uart3m0_xfer: uart3m0-xfer {rockchip,pins =/* uart3_rxm0 */<1 RK_PA0 2 &pcfg_pull_up>,/* uart3_txm0 */<1 RK_PA1 2 &pcfg_pull_up>;};/omit-if-no-ref/uart3m0_ctsn: uart3m0-ctsn {rockchip,pins =/* uart3m0_ctsn */<1 RK_PA3 2 &pcfg_pull_none>;};/omit-if-no-ref/uart3m0_rtsn: uart3m0-rtsn {rockchip,pins =/* uart3m0_rtsn */<1 RK_PA2 2 &pcfg_pull_none>;};/omit-if-no-ref/uart3m1_xfer: uart3m1-xfer {rockchip,pins =/* uart3_rxm1 */<3 RK_PC0 4 &pcfg_pull_up>,/* uart3_txm1 */<3 RK_PB7 4 &pcfg_pull_up>;};};

  接着根据我们的需要继续编写即可,泰山派这里已经默认将uart3配置好了,这里可以直接调用。

//用户串口3
&uart3 {status = "okay";pinctrl-names = "default";pinctrl-0 = <&uart3m1_xfer>;...};

2. 相关API函数

//注册 uart_driver
int uart_register_driver(struct uart_driver *drv)
  • 参数
    • drv :要注册的 uart_driver
  • 返回值
    • 成功:0
    • 失败:负值
//注销uart_driver
void uart_unregister_driver(struct uart_driver *drv)
  • 参数
    • drv :要注销的 uart_driver
  • 返回值:无
//
int serial8250_register_8250_port(struct uart_8250_port *up);
  • 参数
    • up:指向 uart_8250_port 结构体的指针,该结构体包含了要注册的 8250 UART 端口的所有必要信息。
  • 返回值
    • 成功:端口的线路号(line)
    • 失败:错误码
void serial8250_unregister_port(int line);
  • 参数
    • line:要注销的 UART 端口的线路号(line),通常是在调用 serial8250_register_8250_port 时返回的值。
  • 返回值:无

3. 驱动框架

  实际上,我们可以简单地将uart驱动理解为是一个 platform 驱动在驱动入口函数中调用uart_register_driver 函数向 Linux 内核注册 uart_driver,在驱动出口函数中调用uart_unregister_driver 函数注销掉前面注册的 uart_driver,当设备和驱动匹配成功以后进入probe 函数,此函数的重点工作就是初始化uart_8250_port(uart_port),然后将其添加到对应的uart_driver 中。在初始化uart_port过程中,设置 uart_ops 为对应的 XXX_ops即可。

4. 具体功能的实现

4.1 出入口函数的编写

static int __init my_uart_init(void)
{int ret;// 初始化uart_driver结构体my_uart_driver = (struct uart_driver) {.owner      = THIS_MODULE,.driver_name= DRIVER_NAME,.dev_name   = "ttyMY",.major      = TTY_MAJOR,.minor      = 64,.nr         = 1,};// 注册UART驱动ret = uart_register_driver(&my_uart_driver);if (ret)return ret;// 初始化uart_8250_port结构体my_uart_port.port = (struct uart_port) {.iotype     = UPIO_PORT,.mapbase    = UART_BASE,.irq        = UART_IRQ,.uartclk    = UART_CLOCK,.fifosize   = 16,.ops        = &my_uart_ops,.line       = 0,};// 注册8250 UART端口line = serial8250_register_8250_port(&my_uart_port);if (line)uart_unregister_driver(&my_uart_driver);return line;
}static void __exit my_uart_exit(void)
{serial8250_unregister_port(line);uart_unregister_driver(&my_uart_driver);
}module_init(my_uart_init);
module_exit(my_uart_exit);

4.2 读写函数

static void my_uart_start_tx(struct uart_port *port)
{struct uart_8250_port *up = (struct uart_8250_port *)port;unsigned int ier;spin_lock_irq(&up->port.lock);// 启用发送中断ier = serial_in(up, UART_IER);serial_out(up, UART_IER, ier | UART_IER_THRI);spin_unlock_irq(&up->port.lock);
}static void my_uart_stop_tx(struct uart_port *port)
{struct uart_8250_port *up = (struct uart_8250_port *)port;unsigned int ier;spin_lock_irq(&up->port.lock);// 禁用发送中断ier = serial_in(up, UART_IER);serial_out(up, UART_IER, ier & ~UART_IER_THRI);spin_unlock_irq(&up->port.lock);
}static irqreturn_t my_uart_interrupt(int irq, void *dev_id)
{struct uart_8250_port *up = dev_id;unsigned int iir = serial_in(up, UART_IIR);// 检查中断类型并处理if (!(iir & UART_IIR_NO_INT)) {if (iir & UART_IIR_RDI)serial8250_rx_chars(up);if (iir & UART_IIR_THRI)serial8250_tx_chars(up);return IRQ_HANDLED;}return IRQ_NONE;
}

免责声明:本内容部分参考野火科技及其他相关公开资料,若有侵权或者勘误请联系作者。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/31995.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android记录22-关于开发中账号绑定的逻辑处理

有绑定的状态&#xff0c;自然也有取消绑定的状态&#xff1a; 这里的取消绑定只是简单改变应用内的一个状态罢了&#xff0c;并不会跟我们的服务端交互&#xff0c;也不会影响我们的账号登录的状态。 说了这么多&#xff0c;也没看见代码&#xff0c;你们会不会打我&#xff…

vs工程添加属性表

一、简介 1、 vs工程属性表以&#xff08;.props&#xff09;为后缀 2、 作用&#xff1a;当多个工程需要配置很多相同的属性配置时方便同步&#xff0c;比如多个工程需要链接相同的头文件&#xff0c;库文件&#xff0c;输出路径&#xff0c;中间目录等 3、本章内容测试环境&a…

天马学航——智慧教务系统(移动端)开发日志三

天马学航——智慧教务系统(移动端)开发日志三 日志摘要&#xff1a;更新了学生选课模块、我的课程模块以及退课的功能&#xff0c;优化了后端数据库的缓存 1、学生选课模块 学生选课模块主要实现&#xff0c;学生根据需求进行选课操作&#xff0c;通过后端查询到所有教师的课…

Java学习 - 网络IP地址与子网划分 讲解

IP地址 作用 用于决定数据包最终到达哪个计算机 组成 由32位比特组成&#xff0c;即4个字节这32位可以分为两个部分&#xff0c;称为网络号和主机号同一网段的计算机网络号相同&#xff0c;路由器负责连接不同的网段&#xff0c;而交换机负责连接同一网段中不同的计算机同一…

如何将 ChatGPT 集成到你的应用中

在当今快速发展的技术环境中&#xff0c;将人工智能聊天解决方案集成到你的应用程序中可以显著提升用户体验和参与度。OpenAI 的 ChatGPT 以其对话能力和高级语言理解而闻名&#xff0c;对于希望在其应用程序中实现智能聊天功能的开发人员来说是一个绝佳的选择。那我们今天就来…

高性能并行计算华为云实验一:MPI矩阵运算

目录 一、实验目的 二、实验说明 三、实验过程 3.1 创建矩阵乘法源码 3.1.1 实验说明 3.1.2 实验步骤 3.2 创建卷积和池化操作源码 3.2.1 实验说明 3.2.2 实验步骤 3.3 创建Makefile文件并完成编译 3.4 建立主机配置文件与运行监测 四、实验结果与分析 4.1 矩阵乘法…

压缩包文件密码破解软件 Ziperello 下载及使用教程

使用 Ziperello https://qweree.cn/index.php/416/ 对加了密码的压缩包进行密码破解&#xff0c;教程如下&#xff1a; 第一步&#xff0c;双击运行 Ziperello双击我打开程序.exe&#xff0c;如下图&#xff1a; 第二步&#xff0c;打开一个加了密的 ZIP 压缩包&#xff0c;再…

什么概率密度函数?

首先我们来理解一下什么是连续的随机变量&#xff0c;在此之前&#xff0c;我们要先理解什么是随机变量。所谓随机变量就是在一次随机实验中一组可能的值。比如说抛硬币&#xff0c;我们设正面100&#xff0c;反面200&#xff0c;设随机变量为X&#xff0c;那么X{100,200}。 X是…

STM32学习笔记(十)--I2C、IIC总线协议详解

概述&#xff1a;Inter Integrated Circuit&#xff0c;一组多从 多组多从 有应答 是一种同步&#xff08;具有时钟线需要同步时钟SCL&#xff09;、串行&#xff08;一位一位的往一个方向发送&#xff09;、半双工&#xff08;发送接收存在一种&#xff09;通信总线。 &…

【调试笔记-20240618-Windows-pnpm 更新出现 Cannot find module 问题的解决方法】

调试笔记-系列文章目录 调试笔记-20240618-Windows-pnpm 更新出现 Cannot find module 问题的解决方法 文章目录 调试笔记-系列文章目录调试笔记-20240618-Windows-pnpm 更新出现 Cannot find module 问题的解决方法 前言一、调试环境操作系统&#xff1a;Windows 10 专业版调…

qmt量化交易策略小白学习笔记第46期【qmt编程之期货行情数据--如何获取5档盘口行情、期货结算价与持仓量】

qmt编程之获取期货数据 qmt更加详细的教程方法&#xff0c;会持续慢慢梳理。 也可找寻博主的历史文章&#xff0c;搜索关键词查看解决方案 &#xff01; 感谢关注&#xff0c;咨询免费开通量化回测与获取实盘权限&#xff0c;欢迎和博主联系&#xff01; 获取5档盘口行情 …

GeoJson 地图地理信息数据获取

效果图&#xff1a; 获取渠道&#xff1a; 通过阿里数据可视化平台获取通过Vector Maps获取通过geojson来获取 1、通过阿里数据可视化平台获取 2、通过Vector Maps获取 3、通过geojson获取

数据通信与网络(五)

交换机功能&#xff1a; 地址学习&#xff08;端口/MAC地址映射表&#xff09; 通信过滤&#xff08;基于端口/MAC地址映射表&#xff09; 生成树协议&#xff08;断开环路&#xff09; 隔离冲突域 生成树协议 隔离冲突域 交换机配置模式(用不同级别的命令对交换机进行配置) 普…

如何一步一步将Python中的应用打包成安卓的APK安装包文件

一、首先&#xff0c;按照如下链接操作 Python 应用打包成 APK【全流程】_python打包成apk-CSDN博客 二、运行 buildozer init会报错buildozer命令找不到&#xff0c;明明已经安装 解决方法&#xff1a; 这里重新创建一个conda环境 Installation — Buildozer 0.11 docum…

Oracle基本语法(SQLPlus)

目录&#xff1a; 前言&#xff1a; 准备工作&#xff1a; 登录&#xff1a; 1.打开SQL Plus命令行工具 第一种方式&#xff1a; 第二种方式&#xff1a; 2.以不同用户登录 SYSTEM&#xff08;普通管理员&#xff09;&#xff1a; SYS(超级管理员)&#xff1a; 不显示…

408计算机组成原理

todo:有逻辑的分门别类的整理笔记&#xff0c;方便复习 总 理解不了就直接背下来&#xff0c;学越多就越能理解 计算机系统概述 简要目录 基本概念 字长 MAR MDR PC IR CU ALU 通用寄存器、标志寄存器、标志控制器 ACC 地址译码器 通用寄存器 PU C语言编译过程 数据通路带…

DAY10-力扣刷题

1.最后一个单词的长度(简单) 58. 最后一个单词的长度 - 力扣&#xff08;LeetCode&#xff09; 给你一个字符串 s&#xff0c;由若干单词组成&#xff0c;单词前后用一些空格字符隔开。返回字符串中 最后一个 单词的长度。 单词 是指仅由字母组成、不包含任何空格字符的最大子…

Unity3d自定义TCP消息替代UNet实现网络连接

以前使用UNet实现网络连接,Unity2018以后被弃用了。要将以前的老程序升到高版本,最开始打算使用Mirro,结果发现并不好用。那就只能自己写连接了。 1.TCP消息结构 (1). TCP消息是按流传输的,会发生粘包。那么在发射和接收消息时就需要对消息进行打包和解包。如果接收的消息…

Nutch爬虫在大数据采集中的应用案例

引言 在当今信息爆炸的时代&#xff0c;大数据的价值日益凸显。网络作为信息的海洋&#xff0c;蕴藏着丰富的数据资源。Nutch&#xff0c;作为一个开源的Java编写的网络爬虫框架&#xff0c;以其高效的数据采集能力和良好的可扩展性&#xff0c;成为大数据采集的重要工具。本文…

Mac安装多个jdk环境(jdk8+jdk17)保姆级

Mac安装多个jdk环境&#xff08;jdk8jdk17&#xff09;保姆级 背景&#xff1a;新机安装开发环境发现需要找很多文章&#xff0c;&#xff0c;&#xff0c;&#xff0c;这里一篇文章安装所有环境 文章目录 Mac安装多个jdk环境&#xff08;jdk8jdk17&#xff09;保姆级&#x1f…