2021-2022-1 20212825 《Linux内核原理与分析》第五周作业
一、内容学习
1、用户态、内核态和中断
2、系统调用概述
系统调用的意义:操作系统为用户态进程与硬件设备进行交互提供了一组接口——系统调用。把用户从底层的硬件编程中解放出来,极大的提高了系统的安全性,使用户程序具有可移植性。
API和系统调用:API是一个系统调用封装成的一个函数定义;系统调用通过软中断向内核发出一个明确的请求;Libc库定义的一些API引用了封装例程,目的是发布系统调用,让程序员写代码的时候可以通过函数调用而非汇编指令触发一个系统调用;一般每个系统调用对应一个封装例程,库再用这些封装例程定义出给用户的API。
不是每个API都对应一个特定的系统调用。API可能直接提供用户态的服务,比如一些数学函数;一个单独的API可能调用几个系统调用;不同的API可能调用了同一个系统调用。
系统调用的三层皮:xyz(API)、system_call(中断向量)、sys_xyz(中断服务程序)
内核实现了很多不同的系统调用, 进程必须指明需要哪个系统调用,这需要传递一个名为系统调用号的参数(使用eax寄存器来传递)
寄存器传递参数具有如下限制: 1)每个参数的长度不能超过寄存器的长度,即32位 2)在系统调用号(eax)之外,参数的个数不能超过6个(ebx, ecx,edx,esi,edi,ebp)
超过6个怎么办?超过6个的话就把某一个寄存器作为一个指针,指向某一块内存。
API-int 0x80 陷入内核态-systemcall-调用函数
二、实验楼实验
编写test.c程序测试mkdir 系统调用
#include<stdio.h>
#include<stdlib.h>
#include<sys/stat.h>
#include<sys/types.h>
int main(){
int ret;
ret = mkdir("twl",0666);
if(ret == -1)
printf("mkdir twl failed");
else
printf("mkdir twl success!");
return 0;
}
test_asm.c
#include<stdio.h>
#include<stdlib.h>
#include<sys/stat.h>
#include<sys/types.h>
int main(){
int ret;
char *filename="abcd";
int mode=0666;
asm volatile(
"movl %2,%%ecx\n\t"
"movl %1,%%ebx\n\t"
"movl $0x27,%%eax\n\t"
"int $0x80"
:"=a"(ret)
:"b"(filename),"c"(mode)
);
if(ret == -1)
printf("make failed");
else
printf("make success!");
return 0;
}