0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Linux字符设备架构是如何实现的

电子设计 来源:电子设计 作者:电子设计 2020-12-24 18:12 次阅读

一、Linux设备分类

Linux系统为了管理方便,将设备分成三种基本类型:

字符设备块设备网络设备字符设备:

字符(char)设备是个能够像字节流(类似文件)一样被访问的设备,由字符设备驱动程序来实现这种特性。字符设备驱动程序通常至少要实现open、close、read和write的系统调用。

字符终端(/dev/console)和串口(/dev/ttyS0以及类似设备)就是两个字符设备,它们能很好的说明“流”这种抽象概念。

字符设备可以通过文件节点来访问,比如/dev/tty1和/dev/lp0等。这些设备文件和普通文件之间的唯一差别在于对普通文件的访问可以前后移动访问位置,而大多数字符设备是一个只能顺序访问的数据通道。然而,也存在具有数据区特性的字符设备,访问它们时可前后移动访问位置。例如framebuffer就是这样的一个设备,app可以用mmap或lseek访问抓取的整个图像。

在/dev下执行ls -l ,可以看到很多创建好的设备节点:

字符设备文件(类型为c),设备文件是没有文件大小的,取而代之的是两个号码:主设备号5 +次设备号1 。

块设备:

和字符设备类似,块设备也是通过/dev目录下的文件系统节点来访问。块设备(例如磁盘)上能够容纳filesystem。在大多数的Unix系统中,进行I/O操作时块设备每次只能传输一个或多个完整的块,而每块包含512字节(或2的更高次幂字节的数据)。

Linux可以让app像字符设备一样地读写块设备,允许一次传递任意多字节的数据。因此,块设备和字符设备的区别仅仅在于内核内部管理数据的方式,也就是内核及驱动程序之间的软件接口,而这些不同对用户来讲是透明的。在内核中,和字符驱动程序相比,块驱动程序具有完全不同的接口。

块设备文件(类型为b):

网络设备:

任何网络事物都需要经过一个网络接口形成,网络接口是一个能够和其他主机交换数据的设备。接口通常是一个硬件设备,但也可能是个纯软件设备,比如回环(loopback)接口。

网络接口由内核中的网络子系统驱动,负责发送和接收数据包。许多网络连接(尤其是使用TCP协议的连接)是面向流的,但网络设备却围绕数据包的传送和接收而设计。网络驱动程序不需要知道各个连接的相关信息,它只要处理数据包即可。

由于不是面向流的设备,因此将网络接口映射到filesystem中的节点(比如/dev/tty1)比较困难。

Unix访问网络接口的方法仍然是给它们分配一个唯一的名字(比如eth0),但这个名字在filesystem中不存在对应的节点。内核和网络设备驱动程序间的通信,完全不同于内核和字符以及块驱动程序之间的通信,内核调用一套和数据包相关的函数socket,也叫套接字。

查看网络设备使用命令ifconfig:

二、字符设备架构是如何实现的?

在Linux的世界里面一切皆文件,所有的硬件设备操作到应用层都会被抽象成文件的操作。我们知道如果应用层要访问硬件设备,它必定要调用到硬件对应的驱动程序。Linux内核中有那么多驱动程序,应用层怎么才能精确的调用到底层的驱动程序呢?

在这里我们字符设备为例,来看一下应用程序是如何和底层驱动程序关联起来的。必须知道的基础知识:

1.在Linux文件系统中,每个文件都用一个struct inode结构体来描述,这个结构体里面记录了这个文件的所有信息,例如:文件类型,访问权限等。

2.在Linux操作系统中,每个驱动程序在应用层的/dev目录下都会有一个设备文件和它对应,并且该文件会有对应的主设备号和次设备号。

3.在Linux操作系统中,每个驱动程序都要分配一个主设备号,字符设备的设备号保存在struct cdev结构体中。

struct cdev {
struct kobject kobj;
struct module *owner;
const struct file_operations *ops;//接口函数集合
struct list_head list;//内核链表
dev_t dev; //设备号
unsigned int count;//次设备号个数
};
4.在Linux操作系统中,每打开一次文件,Linux操作系统在VFS层都会分配一个struct file结构体来描述打开的这个文件。该结构体用于维护文件打开权限、文件指针偏移值、私有内存地址等信息。

注意:

常常我们认为struct inode描述的是文件的静态信息,即这些信息很少会改变。而struct file描述的是动态信息,即在对文件的操作的时候,struct file里面的信息经常会发生变化。典型的是struct file结构体里面的f_pos(记录当前文件的位移量),每次读写一个普通文件时f_ops的值都会发生改变。

这几个结构体关系如下图所示:

通过上图我们可以知道,如果想访问底层设备,就必须打开对应的设备文件。也就是在这个打开的过程中,Linux内核将应用层和对应的驱动程序关联起来。

1.当open函数打开设备文件时,可以根据设备文件对应的struct inode结构体描述的信息,可以知道接下来要操作的设备类型(字符设备还是块设备)。还会分配一个struct file结构体。

2.根据struct inode结构体里面记录的设备号,可以找到对应的驱动程序。这里以字符设备为例。在Linux操作系统中每个字符设备有一个struct cdev结构体。此结构体描述了字符设备所有的信息,其中最重要一项的就是字符设备的操作函数接口。

3.找到struct cdev结构体后,Linux内核就会将struct cdev结构体所在的内存空间首地记录在struct inode结构体的i_cdev成员中。将struct cdev结构体的中记录的函数操作接口地址记录在struct file结构体的f_op成员中。

4.任务完成,VFS层会给应用层返回一个文件描述符(fd)。这个fd是和struct file结构体对应的。接下来上层的应用程序就可以通过fd来找到strut file,然后在由struct file找到操作字符设备的函数接口了。

三、字符驱动相关函数分析*

* cdev_init() - initialize a cdev structure
* @cdev: the structure to initialize
* @fops: the file_operations for this device

* Initializes @cdev, remembering @fops, making it ready to add to the
* system with cdev_add().

void cdev_init(struct cdev *cdev, const struct file_operations *fops)
功能:
初始化cdev结构体
参数
@cdev cdev结构体地址
@fops 操作字符设备的函数接口地址
返回值:


* register_chrdev_region() - register a range of device numbers
* @from: the first in the desired range of device numbers; must include
* the major number.
* @count: the number of consecutive device numbers required
* @name: the name of the device or driver.

* Return value is zero on success, a negative error code on failure.

int register_chrdev_region(dev_t from, unsigned count, const char *name)
功能:
注册一个范围()的设备号
参数:
@from 设备号
@count 注册的设备个数
@name 设备的名字
返回值:
成功返回0,失败返回错误码(负数)

* cdev_add() - add a char device to the system
* @p: the cdev structure for the device
* @dev: the first device number for which this device is responsible
* @count: the number of consecutive minor numbers corresponding to this
* device

* cdev_add() adds the device represented by @p to the system, making it
* live immediately. A negative error code is returned on failure.

int cdev_add(struct cdev *p, dev_t dev, unsigned count)
功能:
添加一个字符设备到操作系统
参数:
@p cdev结构体地址
@dev 设备号
@count 次设备号个数
返回值:
成功返回0,失败返回错误码(负数)

* cdev_del() - remove a cdev from the system
* @p: the cdev structure to be removed

* cdev_del() removes @p from the system, possibly freeing the structure
* itself.

void cdev_del(struct cdev *p)
功能:
从系统中删除一个字符设备
参数:
@p cdev结构体地址
返回值:

static inline int register_chrdev(unsigned int major, const char *name,
const struct file_operations *fops)
功能:
注册或者分配设备号,并注册fops到cdev结构体,
如果major>0,功能为注册该主设备号,
如果major=0,功能为动态分配主设备号。
参数:
@major : 主设备号
@name : 设备名称,执行 cat /proc/devices显示的名称
@fops : 文件系统的接口指针
返回值
如果major>0 成功返回0,失败返回负的错误码
如果major=0 成功返回主设备号,失败返回负的错误码

该函数实现了对cdev的初始化和注册的封装,所以调用该函数之后就不需要自己操作cdev了。

相对的注销函数为unregister_chrdev

static inline void unregister_chrdev(unsigned int major, const char *name)

审核编辑:符乾江


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Linux
    +关注

    关注

    87

    文章

    10991

    浏览量

    206742
  • 应用层
    +关注

    关注

    0

    文章

    43

    浏览量

    11440
  • Struct
    +关注

    关注

    0

    文章

    29

    浏览量

    10817
收藏 人收藏

    评论

    相关推荐

    ArmSoM系列板卡 嵌入式Linux驱动开发实战指南 之 字符设备驱动

    字符设备驱动 本章,我们将学习字符设备使用、字符设备驱动相关的概念,理解
    的头像 发表于 04-10 09:53 607次阅读
    ArmSoM系列板卡 嵌入式<b class='flag-5'>Linux</b>驱动开发实战指南 之 <b class='flag-5'>字符</b><b class='flag-5'>设备</b>驱动

    linux正则表达式匹配字符

    Linux操作系统中,正则表达式是一种强大的模式匹配工具,它可以用于在文本中查找、替换和筛选特定的字符串。准确掌握正则表达式的语法和常见应用,对于Linux系统的管理员和开发人员来说
    的头像 发表于 11-23 10:26 338次阅读

    linux搜索文件里的字符

    Linux系统中,我们可以使用各种方法来搜索文件中的字符串。在本文中,我将为您详细介绍几种常用的方法和工具。让我们开始吧! grep命令 grep命令是Linux上最常用的搜索工具之一。它可
    的头像 发表于 11-23 10:20 524次阅读

    Linux内核驱动与单个PCI设备的绑定和解绑定

    Linux内核2.6.13-rc3以前,驱动和设备之间的绑定和解绑只能通过insmod(modprobe)和rmmod来实现,但是这种实现方法有一个弊端,就是一旦绑定或者解绑定都是针
    的头像 发表于 11-17 17:11 829次阅读
    <b class='flag-5'>Linux</b>内核驱动与单个PCI<b class='flag-5'>设备</b>的绑定和解绑定

    如何实现一套linux进程间通信的机制

    我们知道linux的进程的间通信的组件有管道,消息队列,socket, 信号量,共享内存等。但是我们如果自己实现一套进程间通信的机制的话,要怎么做?了解android 开发的可能会知道
    的头像 发表于 11-10 14:56 382次阅读
    如何<b class='flag-5'>实现</b>一套<b class='flag-5'>linux</b>进程间通信的机制

    Linux I/O 接口的类型及处理流程

    ()、lseek() 等。 网络 I/O 接口:用于网络通信的接口,包括 socket()、connect()、bind()、listen()、accept() 等。 设备 I/O 接口:用于对设备(e.g. 字符
    的头像 发表于 11-08 16:43 390次阅读
    <b class='flag-5'>Linux</b> I/O 接口的类型及处理流程

    ARM-Linux-IIC设备的添加与驱动实现

    电子发烧友网站提供《ARM-Linux-IIC设备的添加与驱动实现.pdf》资料免费下载
    发表于 10-24 09:55 0次下载
    ARM-<b class='flag-5'>Linux</b>-IIC<b class='flag-5'>设备</b>的添加与驱动<b class='flag-5'>实现</b>

    Emulex HBA CLI ARM架构上的Linux发行说明

    电子发烧友网站提供《Emulex HBA CLI ARM架构上的Linux发行说明.pdf》资料免费下载
    发表于 08-10 16:28 0次下载
    Emulex HBA CLI ARM<b class='flag-5'>架构</b>上的<b class='flag-5'>Linux</b>发行说明

    Linux系统驱动开发之字符设备虚拟设备实验

    本系列图文教程均以全志H3开发板为实验板设计,字符设备驱动开发是最基础的驱动开发。其本质是按字节进行读写操作,读写数据是分先后顺序的。LED、按键、IIC、SPI、USB等等都是字符设备
    发表于 08-07 16:16 332次阅读
    <b class='flag-5'>Linux</b>系统驱动开发之<b class='flag-5'>字符</b><b class='flag-5'>设备</b>虚拟<b class='flag-5'>设备</b>实验

    Linux驱动移植 Linux系统架构优点

    在操作系统上面有应用软件,应用软件程序的执行是依赖于操作系统的,应用程序需要调用 linux 操作系统的库函数来实现,也就是说,应用软件的程序会调用 linux 操作系统的函数来完成对硬件的操作,应用程序是不能对硬件直接进行操
    的头像 发表于 07-27 17:06 570次阅读
    <b class='flag-5'>Linux</b>驱动移植 <b class='flag-5'>Linux</b>系统<b class='flag-5'>架构</b>优点

    架构Arch Linux发行版正式发布

    近日,龙架构Arch Linux发行版官方网站宣布结束beta状态,正式支持龙架构。Arch Linux是一种轻量级、可定制、灵活的Linux
    的头像 发表于 07-18 11:16 613次阅读

    万千设备linux内核如何知道?

    linux内核设备的注册由device_register()函数完成,这个函数是linux设备驱动模型的核心函数
    的头像 发表于 07-12 08:52 506次阅读
    万千<b class='flag-5'>设备</b>,<b class='flag-5'>linux</b>内核如何知道?

    基于Linux使用spidev驱动OLED

    如果不想编写spi设备驱动,那么linux内核提供了一个通用的spidev设备驱动,提供统一的字符设备操作,那么只需要在应用层读写和控制即可
    发表于 06-16 10:36 2731次阅读
    基于<b class='flag-5'>Linux</b>使用spidev驱动OLED

    rtty非常适合远程维护 Linux 设备

    语言实现,前端界面采用 vue 实现。使用 rtty 可以在任何地方通过 Web 访问您的设备的终端,通过 设备ID 来区分您的不同的设备
    的头像 发表于 06-14 09:58 535次阅读

    为什么x86和arm的架构不同,但是都能装linux呢?

    为什么x86和arm的架构不同,但是都能装linux呢?他们的编译时如何实现的?
    发表于 05-16 10:21