2018-2019-1 20189215《Linux内核原理与分析》第五周作业
《庖丁解牛》第四章书本知识总结
- 系统调用的三层机制
API(应用程序编程接口)
中断向量(系统调用处理入口)
服务程序(系统调用内核处理系统)
- 计算机的硬件资源是有限的,为了减少有限资源的访问和使用冲突,CPU和操作系统必须提供一些机制对用户程序进行权限划分,Linux系统采用了Intel x86 CPU的0和3两个特权级别,分别对应内核态和用户态。
- 用户态和内核态很显著的区分方法
CS:EIP的指向范围(对于32位的X86机器):
- 内核态下可以访问0x00000000~0xffffffff的地址空间,共4GB。
- 用户态下只能访问0x00000000~0xbfffffff的地址空间,不能访问从0xc0000000开始的地址空间。
- 一般来说,进入内核态是由中断触发的,系统调用是一种特殊的中断。
- 中断的类别
硬件中断:在用户态进程执行时,硬件中断信号到来,进入内核态,就会执行这个中断对应的中断服务例程。
软中断(trap的方式):用户态程序执行的过程中,调用了一个系统调用,陷入了内核态。
- Linux下系统调用通过int0x80中断完成,中断保存了用户态CS:EIP的值,以及当前的堆栈段寄存器的栈顶,将EFLAGS寄存器的当前值保存到内核堆栈里,同时把当前的中断信号或者是系统调用的终端服务程序的入口加载到CS:EIP中,把当前的堆栈段SS:ESP也加载到CPU里,这些都是由中断信号或者是int指令来完成的。完成后,当前CPU在执行下一条指令时就已经开始执行中断处理程序的入口了,这时对堆栈的操作已经是内核堆栈操作了,
SAVE_ALL
就是内核代码,完成中断服务,如果没有发生进程调度,执行restore_all
恢复中断现场,然后iret
返回到原来的状态;如果发生了进程调度,当前的这些状态就会暂时地保存在系统内核堆栈里,当下一次发生进程调度有机会再切换回当前进程时,执行restore_all
和iret
,这样中断处理过程就执行结束。 - 系统调用通过系统调用号进行区分,通过
EAX
寄存器传递。 - 普通函数调用是通过将参数压栈的方式传递的,而系统调用因为在用户态和内核态下使用不同的堆栈,无法通过参数压栈的方式进行传递,而是通过比较特殊的寄存器传递参数的方式进行。
- 参数按照顺序赋值给
EBX
、ECX
、EDX
、ESI
、EDI
、EBP
,参数的个数不能超过6个,否则的话全部参数应该依次放在一块连续的内存区域里,同时在寄存器EBX
中保存指向该内存区域的指针。
实验:使用库函数API和C代码中嵌入汇编代码两种方式使用同一个系统调用
看了本课程教材之后,我想试试使用超过6个参数的系统调用,可是失败了,具体过程如下:
- 先写了如下代码:
#include <stdio.h>
int main()
{
int ret;
char* n[8] = {"1","2","3","4","5","6","7","8"};
ret = mkdir(n[0],n[1],n[2],n[3],n[4],n[5],n[6],n[7],n[8]);
if(ret == 0)
printf("mkdir successfully!\n");
else
printf("unable to mkdir.\n");
return 0;
}
结果如下:
好像有点不对?试试Linux系统命令:
Linux命令是可以的呀,难道是mkdir系统调用只能传递一个参数吗?
- 编写了嵌入汇编的代码
#include <stdio.h>
int main()
{
int ret;
char* name[8] = {"1","2","3","4","5","6","7","8"};
asm volatile(
"movl %1,%%ebx\n\t"
"movl $0x27,%%eax\n\t"
"int $0x80"
:"=a"(ret)
:"m"(name) //name数组的地址放入内存变量
);
if(ret == 0)
printf("mkdir successfully!\n");
else
printf("unable to mkdir.\n");
return 0;
}
果不其然,结果还是相同:
将代码修改一下,嵌入汇编中的name
替换为name[2]
,重新编译运行,得到如下结果:
从这个结果来看,大约mkdir系统调用的参数无法这样传输多个。
mkdir
系统调用传递两个参数
经过搜索,mkdir
系统调用可以传递的两个参数是文件夹名
和文件夹权限
,因此编写代码如下:
#include <stdio.h>
int main()
{
int ret;
ret = mkdir("1",0755);
if(ret == 0)
printf("mkdir successfully!\n");
else
printf("unable to mkdir.\n");
return 0;
}
权限0755代表只有所有者才有读,写,执行的权限,组群和其他人只有读和执行的权限。编译、运行得到了想要的结果:
- 继续编写嵌入汇编代码
#include <stdio.h>
int main()
{
int ret;
char* name = "1";
int mod = 0755;
asm volatile(
"movl %2,%%ecx\n\t"
"movl %1,%%ebx\n\t"
"movl $0x27,%%eax\n\t"
"int $0x80"
:"=a"(ret)
:"b"(name),"c"(mod)
);
if(ret == 0)
printf("mkdir successfully!\n");
else
printf("unable to mkdir.\n");
return 0;
}
结果相同:
其他想法
本次实验和书本学习,我学到了不少东西,比如mkdir
系统调用的两个参数,再比如查看目录的相关权限可以采用命令ls -lD
,或者直接用ls -la
,更学到了Linux系统中系统调用的三层机制,受益匪浅,也有一个小猜想:Linux系统中的mkdir命令,在添加了多个参数时,是否是通过多次调用mkdir系统调用执行的?还望看到的老师同学们、朋友们能予以解答。
参考资料
2018-2019-1 20189215《Linux内核原理与分析》第五周作业的更多相关文章
- 2019-2020-1 20199329《Linux内核原理与分析》第九周作业
<Linux内核原理与分析>第九周作业 一.本周内容概述: 阐释linux操作系统的整体构架 理解linux系统的一般执行过程和进程调度的时机 理解linux系统的中断和进程上下文切换 二 ...
- 2019-2020-1 20199329《Linux内核原理与分析》第二周作业
<Linux内核原理与分析>第二周作业 一.上周问题总结: 未能及时整理笔记 Linux还需要多用 markdown格式不熟练 发布博客时间超过规定期限 二.本周学习内容: <庖丁解 ...
- 20169212《Linux内核原理与分析》第二周作业
<Linux内核原理与分析>第二周作业 这一周学习了MOOCLinux内核分析的第一讲,计算机是如何工作的?由于本科对相关知识的不熟悉,所以感觉有的知识理解起来了有一定的难度,不过多查查资 ...
- 20169210《Linux内核原理与分析》第二周作业
<Linux内核原理与分析>第二周作业 本周作业分为两部分:第一部分为观看学习视频并完成实验楼实验一:第二部分为看<Linux内核设计与实现>1.2.18章并安装配置内核. 第 ...
- 2018-2019-1 20189221 《Linux内核原理与分析》第九周作业
2018-2019-1 20189221 <Linux内核原理与分析>第九周作业 实验八 理理解进程调度时机跟踪分析进程调度与进程切换的过程 进程调度 进度调度时机: 1.中断处理过程(包 ...
- 2017-2018-1 20179215《Linux内核原理与分析》第二周作业
20179215<Linux内核原理与分析>第二周作业 这一周主要了解了计算机是如何工作的,包括现在存储程序计算机的工作模型.X86汇编指令包括几种内存地址的寻址方式和push.pop.c ...
- 2019-2020-1 20209313《Linux内核原理与分析》第二周作业
2019-2020-1 20209313<Linux内核原理与分析>第二周作业 零.总结 阐明自己对"计算机是如何工作的"理解. 一.myod 步骤 复习c文件处理内容 ...
- 2018-2019-1 20189221《Linux内核原理与分析》第一周作业
Linux内核原理与分析 - 第一周作业 实验1 Linux系统简介 Linux历史 1991 年 10 月,Linus Torvalds想在自己的电脑上运行UNIX,可是 UNIX 的商业版本非常昂 ...
- 《Linux内核原理与分析》第一周作业 20189210
实验一 Linux系统简介 这一节主要学习了Linux的历史,Linux有关的重要人物以及学习Linux的方法,Linux和Windows的区别.其中学到了LInux中的应用程序大都为开源自由的软件, ...
- 2018-2019-1 20189221《Linux内核原理与分析》第二周作业
读书报告 <庖丁解牛Linux内核分析> 第 1 章 计算工作原理 1.1 存储程序计算机工作模型 1.2 x86-32汇编基础 1.3汇编一个简单的C语言程序并分析其汇编指令执行过程 因 ...
随机推荐
- java基础---->Java中图片的缩放
缩略图代表网页上或计算机中图片经压缩方式处理后的小图 ,其中通常会包含指向完整大小的图片的超链接.缩略图用于在 Web 浏览器中更加迅速地装入图形或图片较多的网页.今天,我们就开始java中图像的缩略 ...
- TCP协议的基本规则和在Java中的使用
TCP协议是面向连接的,相对于UDP协议来说效率较低,但是比较安全,数据不容易丢失.TCP协议类似打电话的过程,在一端拨号时必须等待对方回应,确定两端建立了连接通道才能传送信息. 在Java中TCP被 ...
- .net asp 在1.asp页面嵌入另一个页面2.asp
<iframe src="http://www.baidu.com" width="100%" height="100%" onloa ...
- svn-maven-tomcat自动发布脚本
#!/bin/sh #svn-maven-tomcat自动发布脚本 #变量设置 svnpath=svn://10.60.10.120/研发部/xx-maven svnusername=xxx svnp ...
- JS通过正则限制 input 输入框只能输入整数、小数(金额或者现金)
第一: 限制只能是整数 <input type = "text" name= "number" id = 'number' onkeyup= " ...
- PHP 的“魔术常量”
名称 说明 __LINE__ 文件中的当前行号. __FILE__ 文件的完整路径和文件名.如果用在被包含文件中,则返回被包含的文件名.自 PHP 4.0.2 起,__FILE__ 总是包含一个绝对路 ...
- ubuntu系统下用kazam软件录制的视频不能在windows系统下播放的解决方案
遇到问题: 在做计算机视觉课程作业,运动目标检测与跟踪时,在ubuntu系统下用kazam录制了一小段运动目标检测的视频,然后在课上展示时播放不出来,想着Mp4格式的不应该播放不出来啊.网上寻求了一番 ...
- oracle goldengate技术架构-简单试验(全)
一 GoldenGate简介 Oracle Golden Gate软件是一种基于日志的结构化数据复制备份软件,它通过解析源 数据库在线日志或归档日志获得数据的增量变化,再将这些变化应用到目标数据库, ...
- Python将科学计数法数值转换为指定精度浮点数
Python将科学计数法数值转换为指定精度浮点数 In [20]:money = 1190000.0 In [21]: traded_maket_value = 13824000000 In [22] ...
- ScyllaDB - 基础部署
基础环境 操作系统: CentOS 7.2: 集群节点(虚拟机):172.16.134.15 ~ 17: 基础准备 安装依赖和卸载 abrt ( abrt 和 coredump 配置冲突 ): sud ...