Linux内核学习(三)应用层和内核

2023-05-16

目录

  • 写在前面
  • 整体环境
  • 学习笔记
    • 操作系统和内核简介
    • `printf()`和`prinfk()`
    • 应用层对内核的调用
      • 从例子看原理:应用层的`write()`如何调用内核中的`write()`
        • 调用过程实践
        • 实现原理
        • 学习笔记

写在前面

之前做项目的时候,有前辈告诉自己,要去学一下Linux内核,对很多方面都有帮助,现在闲下来,来花时间学一下这一部分的知识点,也算是一个学习笔记
目前跟着B站UP主——简说linux 的教程《Linux内核开发100讲》学习,链接如下:
简说linux个人空间
本章学习参考链接:
printk和printf的区别
《Linux内核设计与实现》
在学习的过程中,我也会对遇到的各种问题进行深一步学习, 从而总结知识点到博客当中,这就会出现内容可能会四处跳跃,但是这种跳跃符合我的学习过程。

整体环境

为了学习代码,我们需要一个一套Linux环境,因为为了方便自己记笔记和学习,没有用双系统,直接在windows10下面用VMware建了一个虚拟机进行试验。
开发环境:VMWare虚拟机 Ubuntu 18.04
Linux源码版本:linux4.9.229

学习笔记

这一章是关于Linux内核的一个总体印象,以及应用层和驱动层之间相互调用的逻辑关系。

操作系统和内核简介

在《Linux内核设计与实现》中认为,操作系统是指在整个系统中负责完成最基本功能和系统管理的那些部分。这些部分应该包括内核、设备驱动程序、启动引导程序、命令行Shell或者其他种类的用户界面、基本的文件管理工具和系统工具。
而内核时操作系统的核心所在,系统的其他部分都必须依靠内核这部分软件提供的服务,比如管理硬件设备、分配系统资源等等。

内核组成
负责响应中断的中断服务程序
负责管理多个进程从而分享处理器时间的调度程序
负责管理进程地址空间的内存管理程序
网络,进程间通信等服务程序

对于提供保护机制的现代系统来说,内核独立于普通应用程序,一般处于系统态,拥有受保护的内存空间和访问硬件设备的所有权限,这种系统态被保护起来的内存空间,统称为内核空间。
而与这个相对的是应用程序在用户空间执行。只能看到允许它们使用的部分系统资源,不能直接访问硬件,也不能访问其他人的内存范围。执行一个应用程序的时候,系统将以用户态进入用户空间执行。
系统中运行的应用程序通过系统调用来实现与内核通信。具体的图如下:
在这里插入图片描述
应用程序通常调用库函数(比如C库函数)再由库函数通过系统调用界面,让内核代其完成各种不同任务。一些库调用提供了系统调用不具备的许多功能。

printf()prinfk()

在Linux内核学习(二)里面,我们就在我们自己的设备驱动里面用了printk()函数来进行信息的输出从而进行调试,但为啥不用C语言里面的printf函数呢?
因为大部分的C语言库里面的函数在内核中都得到了实现,但是printf()函数是并没有被实现的,因此在我们前面的内核驱动设备的C语言代码中,我们是无法调用printf()函数的,但是没有了这个函数,内核中使用了另一个函数,printk()

  • printk基本上在任何时候和任何地方都能够调用,它的使用弹性极佳
  • printk可以指定一个日志警告级别,内核可以根据这个警告级别来判断是否需要在终端上打印信息。内核会把级别比某个特定值低的所有消息显示在终端上。记录等级如下:
记录等级描述记录等级
KERN_EMEG一个紧急情况0
KERN_ALERT一个需要立即被注意到的错误1
KERN_CRIT一个临界情况2
KERN_ERR一个错误3
KERN_WARNING一个警告4
KERN_NOTICE一个普通的,不过也有可能需要被注意的情况5
KERN_INFO一条非正式的消息6
KERN_DEBUG一条调试信息——一般是冗余信息7

如果没有特别指定一个记录等级,函数会选用默认的DEFAULT_MESSAGE_LOGLEVEL,默认等级是KERN_WARNING,内核里面最重要的记录等级是KERN_EMEG,按照表格从上往下对应从重要到不重要的记录等级,最无关紧要的是KERN_DEBUG。当记录等级低于默认等级的时候,不会在终端里面显示,而显示在日志里面
sudo dmesg 查看日志信息
sudo dmesg -C 清除日志信息
我们可以查看当前的记录等级

cat /proc/sys/kernel/printk

我们试着输入一下:返回了 4 4 1 7

cat /proc/sys/kernel/printk
4	4	1	7

输出结果中的四个数字分别代表当前记录等级默认等级最小记录等级,和最大记录等级
由前面得知,低于默认等级的时候,不会在终端显示,而在日志显示,因此等级0-3会输出到终端,4-7只会显示在日志当中.
我们可以使用sudo echo "6" > /proc/sys/kernel/printk 来改变系统的默认等级。注:此改变方法,可能需要使用sudo su切换到root用户才可以修改。

应用层对内核的调用

在我前面写的Linux内核学习(二)里面,我们已经实现了一个设备驱动的插入,并且其设备具有三个操作,open,write,read。那当我们有了一个设备之后,我们就可以在我们应用层通过系统调用的接口调用设备驱动,从而实现对硬件设备的调用。当然,由于我们是没有具体的硬件的,所以我们省略最后一步设备驱动对硬件设备的调用,只了解应用层如果实现对内核里面的设备驱动的调用

从例子看原理:应用层的write()如何调用内核中的write()

调用过程实践

要实现这个例子,我们当然首先要手写一个应用层的程序test.c,随便放到哪,我把学习过程中的笔记按照注释写在代码中,具体如下:

#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <sys/select.h>

#define DATA_NUM (64)
int main(int argc, char *argv[])
{
    int fd,i;
    int r_len,w_len;
    fd_set fdset;
    char buf[DATA_NUM]="hello world"; //创建一个字符缓冲数组,以便后续使用
    memset(buf,0,DATA_NUM); //将DATA_NUM中剩余的值填充到buf中
    fd = open("/dev/hello", O_RDWR); //用可读写的权限打开设备驱动hello
    printf("%d\r\n", fd);			
    if(-1 == fd)				//判断文件是否打开
    {
        perror("open file error\r\n");
		return -1;	
    } 
    else
    {
		printf("open successe\r\n");
    }	
    w_len = write(fd, buf, DATA_NUM);	//打开成功 就调用write和read
    r_len = read(fd, buf, DATA_NUM);
    printf("%d%d\r\n", w_len, r_len);	//将返回值打印出来
    printf("%s\r\n",buf);

    return 0;
}

然后我们来运行这个运行这个程序
我们先按照最基本的C语言的流程,编译它。

gcc -o test test.c

然后我们运行它

./test

但我们发现它无法打开文件,返回以下错误

-1
open file error
: No such file or directory

其原因是,虽然我们在内核里面注册了我们的内核驱动,但是我们在应用层里面没有建立这样的一个文设备件。虽然在现在的Linux内核可以自动生成这样的设备文件,但UP主给我们演示了具体的实现过程:
首先,我们需要创建一个设备文件,需要使用mknod命令,其用法如下:
mknod [OPTION ]NAME TYPE [MAJOR MINOR]
TYPE是设备的类型,MAJOR和MINOR指的是主设备号和次设备号

mknod /dev/hello c 232 0
# 由于我们应用层中写的打开设备时hello,所以这里名字和代码中的文件名一样
# c 代表着这是一个字符设备
# 232 0 是我们上一节写的内核注册的驱动文件中的主次设备号
ls -l /dev/hello
# 此时再用ls命令就可以看到返回的设备文件啦

在这里插入图片描述
然后我们再清空一下我们的日志,并执行测试程序,

sudo dmesg -C
./test

发现返回了以下数据
在这里插入图片描述
这样,我们就运行完了了我们应用层的软件代码了,上述的过程中,我们是调用了我们前面写的驱动来进行的,而在我们前一节的笔记中,刚好在内核的驱动文件中,有三个对应应用层软件代码中open,write,read的函数,并对其进行了指向具体。代码如下:

int hello_open(struct inode *p, struct file *f)
{
    printk(KERN_EMERG"hello_open\r\n");
    return 0;
}

ssize_t hello_write(struct file *f, const char __user *u, size_t s, loff_t *l)
{
    printk(KERN_EMERG"hello_write\r\n");
    return 0;
}
ssize_t hello_read(struct file *f, char __user *u, size_t s, loff_t *l)
{
    printk(KERN_EMERG"hello_read\r\n");      
    return 0;
}
ps:函数里面的loff_t是一个类型的声明,其本质就是一个long long类型

在这里插入图片描述
这个时候,我们再使用dmesg查看日志的时候,发现这个应用层确实调用了驱动层中的驱动设备,并输出了对应的日志内容,那么具体是怎么实现的呢?
在这里插入图片描述

实现原理

由于用户空间的程序是无法直接执行内核代码。它们不能直接调用内核空间中的函数,因为内核驻留在受保护的地址空间。如果进程可以直接在内核的地址空间上读写的话,系统的安全系和稳定性将不复存在
因此,应用程序需要通过某种方式通知系统,告诉内核自己需要执行一个系统调用,希望系统切换到内核态,这样内核就可以代表应用程序在内核空间执行系统调用。
通知内核的机制是靠软中断来实现的,通过引发一个异常来促使系统切换到内核态去执行异常处理程序。这个异常处理程序实质上就是一个系统调用处理程序。处理的程序就叫做system_call()

具体的过程用图片表示如下:
在这里插入图片描述
以之前的应用层的write()为例,具体的过程如下:
首先应用层的代码执行之后,产生一个中断,然后被系统调用程序进行处理。具体的就是一个SYSCALL_DEFINE3(write, unsigned int, fd, const char __user *, buf,size_t, count)这样的一个函数
然后这个调用会将其调用到ssize_t __vfs_write(struct file *file, const char __user *p, size_t count, loff_t *pos)的函数,就是在这个函数中,将我们设备驱动中的read,open,write三个函数给调用了。
最终逐层传递回参数给用户空间,从而实现了整个过程

学习笔记

在上述的test.c文件中,有以下几个需要注意的点

  • 我们在上面的test.c文件中调用的是printf()函数,而不是printk(),原因是这个c文件是一个应用层的软件,而不是内核中的驱动文件,所以他是可以直接调用C库中的函数,并打印到终端中。
  • 我们打开设备的操作是open(),这是因为Linux系统中,把所有的东西都文件化了,因此,一个设备也是一个文件,而要使用一个文件,第一步当然是先打开这个文件啦
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Linux内核学习(三)应用层和内核 的相关文章

  • IDEA快捷创建Servlet项目

    IDEA快捷创建Servlet项目 1 右键单击 要创建Servlet项目 所在的包 注 xff1a 若没有Servlet选项 打开File gt project structure 进入如下页面 xff0c 选择你使用的工程 xff0c
  • 【基础教程】Python转义字符及用法

    ASCII 编码为每个字符都分配了唯一的编号 xff0c 称为编码值 在 Python 中 xff0c 一个 ASCII 字符除了可以用它的实体 xff08 也就是真正的字符 xff09 表示 xff0c 还可以用它的编码值表示 这种使用编
  • SAP 实施新的金融工具 IFRS17规则解析

    在实施新的金融工具 IFRS 规则的过程中 xff0c 保险公司现在看到了保险负债的新标准 经过多年的长期讨论 xff0c IASB 于 2016 年 11 月承诺在 2021 年 1 月 1 日生效 xff0c 并明确表示不会考虑进一步推
  • FreeRTOS中任务控制块中关于堆栈的三个变量pxTopOfStack、pxStack、pxEndOfStack的分析

    这里写自定义目录标题 FreeRTOS中任务控制块中关于堆栈的定义 typedef struct tskTaskControlBlock volatile StackType t pxTopOfStack lt Points to the
  • 在keil中添加c文件时的一些常见错误

    近期在做项目时 xff0c 计划添加一个测距模块 xff0c 因此要新建一个c文件封装一些函数 xff0c 方便调用 xff0c 在实际操作时遇到了以下两个问题 1 编译时报错 error no such file or dictionar
  • stm32f1驱动HC-SR04超声波测距模块

    这段时间算是比较系统的学习了一点stm32相关的东西 xff0c 驱动超声波测距模块这个简单的小任务综合了定时器 xff0c 中断 xff0c 串口三个知识点 xff0c 拿来练手还是挺不错的 首先先介绍一下HC SR04模块 xff1a
  • HC05/HC06蓝牙模块驱动(1)

    花了点时间熟悉了一下蓝牙模块 xff0c 发现意料之外的简单 先说蓝牙模块的三种工作模式 xff08 这三种工作模式是HC05 06共有的 xff0c 并且通过模块上的LED反映出来 xff09 1 待连接模式 HC05 快速闪灯 HC06
  • Raft的PreVote实现机制

    Raft的PreVote实现机制 1 背景 在Basic Raft算法中 xff0c 当一个Follower与其他节点网络隔离 xff0c 如下图所示 xff1a Follower 2在electionTimeout没收到心跳之后 会发起选
  • ubuntu环境下opencv学习+踩坑

    opencv学习 43 踩坑 环境 ubuntu 19 04vscode 1 37 0opencv 3 4 7cmake 3 13 4 自己的blog在持续更新中 https blog csdn net sasasatori article
  • jetson nano上opencv+vscode环境搭建+踩坑记录

    jetson nano踩坑 1 英伟达系的cpu是arm64架构 xff0c 而市面上大部分直接支持的软件是amd64架构 xff0c 这就导致了很多很好用的软件无法直接在上面用 xff0c 比如simple note 2 vsc我是找了个
  • rc视觉工作日常

    RC视觉 由于硬件条件限制 xff0c 最后选择的开发方案为主机 43 jetson nano交叉开发的方案 xff0c 计划实现的目标为识别到某个特征物体 xff08 比如运动的羽毛球 xff09 之后 xff0c 将处理之后的消息通过串
  • 服务器后台语言选型参考

    1 后台服务器什么技术搭建 xff1f 2 为什么大型网站前端使用 PHP 后台逻辑用 Java xff1f 3 为什么大型网站前端使用 PHP 后台逻辑用 Java xff1f 4 Java在Web开发语言上败给了PHP
  • MVC4 网站发布(整理 + 部分转载 + 部分问题收集和解决方案)

    网站发布步骤 这部分是转载文章 在此标明出处 xff0c 以前有文章是转的没标明的请谅解 xff0c 因为有些已经无法找到出处 xff0c 或者与其它原因 如有冒犯请联系本人 xff0c 或删除 xff0c 或标明出处 因为好的文章 xff
  • 2022-适用于 Windows 10 Version 1809 的 02 累积更新,适合基于 x64 的系统 (KB5010351) - 错误 0x800f0982

    2022 适用于 Windows 10 Version 1809 的 02 累积更新 xff0c 适合基于 x64 的系统 KB5010351 错误 0x800f0982 系统是win10 企业版 LTSC版本 可能安装的是精简版导致的 运
  • sqlsever中text字段类型是否会影响查询性能

    先上结论 会影响查询性能 我在库里找了一张表T Sys Log 然后做2个副本 备份表 SELECT INTO T Sys Log back FROM T Sys Log SELECT INTO T Sys Log back2 FROM T
  • 【无标题】

    起因 2010 年 xff0c 谷歌宣布退出地内市场的时候 xff0c 一直保留着 谷歌地图 和 谷歌翻译 这两个公共服务 有兴趣自行百度下谷歌和百度恩怨 在 2020 年 xff0c 谷歌停止了 谷歌地图 在内地的服务 现在 xff0c
  • vmware ESXI 裸金属架构 本地服务器 开启Intel VT-x(虚拟化技术)

    我想使用vmware ESXI 安装的WIN10虚拟机中装vmware软件再装win10 即虚拟机中套虚拟机 基于工作要求某个XXX项目需要开启VPN远程到客户内网进行维护 客户对网络审计比较严 安装VPN的那台机子识别码要上传服务器 基本
  • 对一个或多个实体的验证失败。有关详细信息,请参见“EntityValidationErrors”属性。

    因为是转载文章 在此标明出处 xff0c 以前有文章是转的没标明的请谅解 xff0c 因为有些已经无法找到出处 xff0c 或者与其它原因 如有冒犯请联系本人 xff0c 或删除 xff0c 或标明出处 因为好的文章 xff0c 以前只想收
  • leveldb之log文件

    leveldb之log文件 1 log文件在LevelDb中的主要作用是系统故障恢复时 xff0c 能够保证不会丢失数据 因为在将记录写入内存的Memtable之前 xff0c 会先写入Log文件 xff0c 这样即使系统发生故障 xff0
  • STM32单片机的CRL和CRH寄存器

    这里写目录标题 问题 xff1a 基础知识 xff1a 解释 xff1a 扩展PA1为输入 上 下拉 PA2为输入 上 下拉 PA1为输出 xff08 通用推挽输出50MHZ xff09 PA2为输出 xff08 通用推挽输出50MHZ x

随机推荐

  • visualsvn server 无法访问url

    IIS 发布网站 本机能访问 其它人访问不了 看一下服务端 VisualSVN Server 的服务有没有启动 x A 34 H g6 L N s 管理 服务 VisualSVN Server 备注 做为开发机子 手动优化自己的电脑吧 否则
  • JS日期加减,日期运算

    因为是转载文章 在此标明出处 xff0c 以前有文章是转的没标明的请谅解 xff0c 因为有些已经无法找到出处 xff0c 或者与其它原因 如有冒犯请联系本人 xff0c 或删除 xff0c 或标明出处 因为好的文章 xff0c 以前只想收
  • jQuery easyui 选中特定的tab

    获取选中的 Tab 1 获取选中的 tab panel 和它的 tab 对象 2 var pp 61 39 tt 39 tabs 39 getSelected 39 3 var tab 61 pp panel 39 options 39 t
  • Server Error in '/' Application. 解决办法

    Server Error in 39 39 Application Access to the path 39 E NetWeb2 Content upFile BClientExcel 大客户部通讯录导入 xlsx 39 is denie
  • easyui-datagrid 数据出不来(样式引起的bug)

    今天任务是需要从另一个项目中将某几个功能页面移植到现有的项目中 这是比较繁琐的功能 理解要移植功能的逻辑 xff08 业务逻辑 xff0c 涉及到的表和存储过程 xff09 页面样式 这么是我遇到的一个问题之一 xff1b 我需要展现一个e
  • c#切割字符串几种方法

    1 xff0c 按单一字符切割 string s 61 34 abcdeabcdeabcde 34 string sArray 61 s Split 34 c 34 oreach string i in sArray Console Wri
  • Linux

    一 常用操作以及概念 快捷键求助关机PATHsudo包管理工具发行版VIM 三个模式GNU开源协议 二 磁盘 磁盘接口磁盘的文件名 三 分区 分区表开机检测程序 四 文件系统 分区与文件系统组成文件读取磁盘碎片blockinode目录日志挂
  • 动态链接库与静态链接库的区别

    静态链接库与动态链接库都是共享代码的方式 xff0c 如果采用静态链接库 xff0c 则无论你愿不愿意 xff0c lib 中的指令都全部被直接包含在最终生成的 EXE 文件中了 但是若使用 DLL xff0c 该 DLL 不必被包含在最终
  • ssm——小学期实训总结

    实训总结 经过这两个星期短暂的学习 xff0c 我学习了ssm的框架搭建与web前端设计基础 在第一个星期 xff0c 老师着重为我们讲了框架的原理 搭建与运用 xff1b 而在第二个星期 xff0c 重点则转移到了小组对项目的开发与研究上
  • ROS TF原理和使用方法

    ROS TF介绍 一 TF是什么 xff1f 1 TF是ROS的一个包 xff08 package xff09 2 TF能让用户随时记录各种坐标系之间的变换关系 3 TF能让用户在一个坐标系中进行坐标运算 xff0c 并将转换关系后的位置关
  • 分布式系统核心—日志

    分布式系统的核心组件 日志 有时也叫write ahead logs commit logs 或者事物 logs 通常指在应用所有的修改之前先写入日志 xff0c 一般会将重放日志 撤销日志都写进去 NoSQL数据库 KV存储 Hadoop
  • Linux 下常见的进程调度算法

    进程调度 xff1a 在操作系统中调度是指一种资源分配 调度算法是指 根据系统的资源分配策略所规定的资源分配算法 操作系统管理了系统的有限资源 xff0c 当有多个进程 或多个进程发出的请求 要使用这些资源时 xff0c 因为资源的有限性
  • Ubuntu18.04更换内核方法(原内核版本 4.15.0-38-generic)

    以下过程全部在root权限下操作 xff08 sudo su xff09 1 安装必备软件编译工具 xff1a apt get install libncurses5 dev build essential kernel package 注
  • Mac下使用Java反编译工具JD-GUI

    下载 下载JD GUI 我们选择 Mac 版的 jd gui osx 1 6 6 tar 下载解压打开即可使用 xff0c 不出意外的话出意外了 竟然提示我没有找到Java 版本 xff0c 我直接zsh 命令行下执行查看 java ver
  • Android 中使用Lambda表达式

    Android Studio默认使用Lambda表达式是会报错的 xff0c 即使你使用的是java 8 xff0c 为了在android studio中使用lambda表达式 xff0c 我们必须借助一个插件retrolambda xff
  • 树莓派安装ros系统

    导语 最近给树莓派安装了ros系统 xff0c 这里记录一下 步骤 xff1a 1 下载ros系统的软件 这里推荐从ubiquityrobotics下载ubiquityrobotics 的系统 这个相当于是给你下载了ubuntu16 04和
  • 嵌入式软件工程师相关的应聘要求

    本文收集从网上找到的嵌入式软件工程师岗位相关的职位要求 xff0c 与自身能力进行对比 xff0c 找出不足 xff0c 查漏补缺 xff0c 为18年的跳槽做好准备 1 嵌入式软件工程师杭州 浙江大华技术股份有限公司 职位描述 xff1a
  • docker无法访问localhost的一种解决方法

    如果你使用的不是toolbox xff0c 可以关掉这个页面了 如果你使用的是toolbox xff0c 请使用192 168 99 100加你的的接口 因为toolbox使用了virtualbox虚拟机 xff0c 相当于包了一层 xff
  • VCC、VDD、VEE、VSS等有关电源标注的区别

    Almost all integrated circuits ICs have at least two pins which connect to the power rails of the circuit they are insta
  • Linux内核学习(三)应用层和内核

    目录 写在前面整体环境学习笔记操作系统和内核简介 96 printf 96 和 96 prinfk 96 应用层对内核的调用从例子看原理 应用层的 96 write 96 如何调用内核中的 96 write 96 调用过程实践实现原理学习笔