linux内核分析学习笔记 ——第五章 系统调用的三层机制

学习重点——深入理解系统调用的过程


给MenuOS添加命令

添加命令的方式较为简单,在LinuxKernel/menu/test.c目录下,打开test.c,main函数中的MenuConfig就是对应的系统调用的功能增加,之后在上面给出具体实现即可。之后就可以在MenuOS中使用该系统调用函数。

使用gdb跟踪内核函数sys_getuid

上面在MenuOS中添加了功能,我这里改成了getuid获得当前用户的id号,调试步骤和实验三基本一样

qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img -S -s

冻结内核运行,然后利用gdb调试工具,加载内核,连接端口1234

file linux-3.18.5/vmlinux

target remote:1234

之后在系统调用sys_getuid处设置断点,当启动MenuOS调用功能Getuid后 一旦调用系统函数,程序暂停。可以看到程序停在了sys_getuid处,但该系统调用是利用宏函数实现的,经过预编译处理后是sys_getuid的形式。

system_call位置设置断点并不能停下,还是会停在sys_getuid这里,因为system_call并不是一个正常的函数,只是一段汇编代码的起点,内部没有严格的函数调用堆栈机制,所以gdb不能完成跟踪任务

系统调用的处理过程

在用户态中有一个c语言提供的APIxyz(),在用户看来这就是系统调用,而xyz()中调用了SYSCALL来触发系统调用。

即中断向量0x80对应system_call中断服务程序入口。

start_kernel函数执行内核启动的初始化工作,其中会调用trap_init函数,在上图所示目录中可以看到以下代码。

当接受到int 0x80 中断请求后,通过set_sysytem_trap_gate函数将中断向量和入口函数绑定,将直接进入中断,也就是跳转到system_call的位置。

这里将trap_init中绑定中断向量和入口函数的代码中的宏定义实现,可以看到系统调用对应的中断向量就是0x80

system_call函数的理解

`system_call`是一段中断服务程序入口的汇编代码,就是系统调用的处理过程,是系统调用用户态和系统调用内核态想转换的过程。

中断过程都会有现场保护和恢复现场代码一开始的SAVE_ALL和restore_all就是实现了中断上下文过程。

sys_call_table是一个系统调用表,EAX寄存器保存了传递的系统调用号,来调用相应的系统调用,调用结束,EAX寄存器还要保存系统调用的返回值。

syscall_exit中判断当前的任务是否需要进程调度,如果需要进程调度进入syscall_exit_work,执行进程调度,结束后恢复现场返回用户态。

系统调用函数

上图所示是系统调用system_call的流程图

  • SAVE ALL 是中断发生后保存现场的宏
  • cmpl $(nr_syscalls),%eax判断检查系统调用号的合法性如果系统调用号大于nr_syscalls 那么出现异常,跳入异常处理syscall_badsys

    - call *sys_call_table(,%eax,4)通过系统调用号在系统调用表中找到相应的系统调用内核处理函数,就是将函数API和系统调用联系起来。
  • movl %eax,PT_EAX(%esp)作用是将系统调用返回值入栈
  • syscall exit检查是否有任务要处理,如果发生进程调度需要进入syscall_exit_work
    • 进程间通信会有信号发生,所以进入进程调度后会有work_pendingwork_notifysig信号处理、和work_resched等处理 work_resched中的schedule存着进程调度的代码,也就是进程调度的时机点
  • restore_alliret用于恢复现场和返回用户态。

Linux内核编译实践中遇到的问题

  • 在为函数添加系统调用号时的路径问题

    sudo gedit /usr/src/linux-3.14.40/arch/x86/syscall/syscall_32.tbl。

    根据文档我在这个路径下并没有找到syscall文件夹,也就无法修改 syscall_32.tbl

最后参考了一篇博客,我想问题应该是内核版本更新,目录有了一定的修改,最后找到了路径

  • 使用 make bzImage时出现错误

根据错误提示,我考虑问题出在了内核调用时添加的系统调用格式出现了问题对比之后,我对代码进行更改,结果编译成功

内核编译时间非常久,编译好了之后无法启动,错误问题如下

后来发现是虚拟机的内存不足的原因,重新设置了虚拟机的内存之后成功进入虚拟机,利用命令可以看到该内核版本已经是刚刚编译的4.18.18

写一个程序调用我们刚刚创建的系统调用,返回值得到的是我们当时设定的返回值

参考这篇博客给我提供了解决问题的很多灵感:

Linux4.18.9添加系统调用传递参数示例

2018-2019-1 20189206 《Linux内核原理与分析》第六周作业的更多相关文章

  1. 2019-2020-1 20199329《Linux内核原理与分析》第九周作业

    <Linux内核原理与分析>第九周作业 一.本周内容概述: 阐释linux操作系统的整体构架 理解linux系统的一般执行过程和进程调度的时机 理解linux系统的中断和进程上下文切换 二 ...

  2. 2019-2020-1 20199329《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 一.上周问题总结: 未能及时整理笔记 Linux还需要多用 markdown格式不熟练 发布博客时间超过规定期限 二.本周学习内容: <庖丁解 ...

  3. 20169212《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 这一周学习了MOOCLinux内核分析的第一讲,计算机是如何工作的?由于本科对相关知识的不熟悉,所以感觉有的知识理解起来了有一定的难度,不过多查查资 ...

  4. 20169210《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 本周作业分为两部分:第一部分为观看学习视频并完成实验楼实验一:第二部分为看<Linux内核设计与实现>1.2.18章并安装配置内核. 第 ...

  5. 2018-2019-1 20189221 《Linux内核原理与分析》第九周作业

    2018-2019-1 20189221 <Linux内核原理与分析>第九周作业 实验八 理理解进程调度时机跟踪分析进程调度与进程切换的过程 进程调度 进度调度时机: 1.中断处理过程(包 ...

  6. 2017-2018-1 20179215《Linux内核原理与分析》第二周作业

    20179215<Linux内核原理与分析>第二周作业 这一周主要了解了计算机是如何工作的,包括现在存储程序计算机的工作模型.X86汇编指令包括几种内存地址的寻址方式和push.pop.c ...

  7. 2019-2020-1 20209313《Linux内核原理与分析》第二周作业

    2019-2020-1 20209313<Linux内核原理与分析>第二周作业 零.总结 阐明自己对"计算机是如何工作的"理解. 一.myod 步骤 复习c文件处理内容 ...

  8. 2018-2019-1 20189221《Linux内核原理与分析》第一周作业

    Linux内核原理与分析 - 第一周作业 实验1 Linux系统简介 Linux历史 1991 年 10 月,Linus Torvalds想在自己的电脑上运行UNIX,可是 UNIX 的商业版本非常昂 ...

  9. 《Linux内核原理与分析》第一周作业 20189210

    实验一 Linux系统简介 这一节主要学习了Linux的历史,Linux有关的重要人物以及学习Linux的方法,Linux和Windows的区别.其中学到了LInux中的应用程序大都为开源自由的软件, ...

  10. 2018-2019-1 20189221《Linux内核原理与分析》第二周作业

    读书报告 <庖丁解牛Linux内核分析> 第 1 章 计算工作原理 1.1 存储程序计算机工作模型 1.2 x86-32汇编基础 1.3汇编一个简单的C语言程序并分析其汇编指令执行过程 因 ...

随机推荐

  1. MySQL操作数据库--与MySQL零距离接触1-7

    第一章 1-7操作数据库 数据库是一个集合:表 索引等. MySQL语句规范: 关键字与函数名称全部大写 数据库名称.表名称.字段名称全部小写 SQL语句必须以分号结尾 语法结构:       {}: ...

  2. 编译用到boost相关的东西,问题的解决;以及和googletest库

    编译https://github.com/RAttab/reflect, 发现需要gcc4.7以上的版本才行.于是编译安装最新的gcc-6.2.0, 过程算顺利. http://www.linuxfr ...

  3. linux安装svn客户端rabbitvcs

    我们都知道,自从svn出道以来,很多人都预言,cvs将会被其取代.就如同他们预言maven要取代ant一样.可见,svn的流行.在Windows中,最常用到的开源免费的svn客户端就是Tortoise ...

  4. cocos2d-x android工程接入第三方支付宝SDK

    1. 首先去支付宝官网下载开发者文档 2. 然后按着开发者文档将支付宝的sdk导入到你的工程中,并关联到工程中,步骤入下图: (1)将从支付宝官方网站获得的支付宝的sdk的jar包拷贝到工程中的lib ...

  5. opencv之模糊处理

    初学OpenCV的开发者很容易被OpenCV中各种滤波方法所困扰,不知道到底该用哪里一个来做滤波.表面原因看起来是因为OpenCV中各种滤波方式实在是太多太杂, 其背后原因是对各种滤波方法的应用场景认 ...

  6. LeetCode13.罗马数字转整数

    罗马数字包含以下七种字符: I, V, X, L,C,D 和 M. 字符 数值 I 1 V 5 X 10 L 50 C 100 D 500 M 1000 例如, 罗马数字 2 写做 II ,即为两个并 ...

  7. IIS 8.0 Using ASP.NET 3.5 and ASP.NET 4.5微软官方安装指导

    from:https://www.iis.net/learn/get-started/whats-new-in-iis-8/iis-80-using-aspnet-35-and-aspnet-45 S ...

  8. 解决lnmp服务器环境wordpress升级或安装插件主题要输入FTP的方法

    使用lnmp一键安装包搭建的服务器环境运行wordpress,在后台升级WP程序.安装或更新插件主题时,每时提示输入FTP信息,该问题的原因是nginx的执行身份用户组非文件用户组,前面博客吧使用虚拟 ...

  9. 转 VS2010 RDLC 横向合并时“未正确设置 tablix“Tablix1”的 FixedData 属性”错误解决方法 .

    最近在使用Rdlc做报表打印,有些报表的表头需要合并表头.Rdlc本身提供了横向合并的工具,但是在实际合并的时候,会出现“未正确设置 tablix“Tablix1”的 FixedData 属性.除非在 ...

  10. codeforces 980D Perfect Groups

    题意: 有这样一个问题,给出一个数组,把里面的数字分组,使得每一个组里面的数两两相乘都是完全平方数. 问最少可以分成的组数k是多少. 现在一个人有一个数组,他想知道这个数组的连续子数组中,使得上面的问 ...